<!DOCTYPE html>
<html lang="en">
	<head>
		<title>games - main.go</title>
		<link type="text/css" rel="stylesheet" href="/style/src.css">
	</head>
	<body>
		<h1><a href="/games">games</a> - main.go</h1>
		<pre>
<span class="hidden"><a id="L1" href="#L1">     1</a>  </span><span>package main</span>
<span class="hidden"><a id="L2" href="#L2">     2</a>  </span>
<span class="hidden"><a id="L3" href="#L3">     3</a>  </span><span>import (</span>
<span class="hidden"><a id="L4" href="#L4">     4</a>  </span><span>	&#34;encoding</span><span>/base64&#34;</span>
<span class="hidden"><a id="L5" href="#L5">     5</a>  </span><span>	&#34;flag&#34;</span>
<span class="hidden"><a id="L6" href="#L6">     6</a>  </span><span>	&#34;fmt&#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;strings&#34;</span>
<span class="hidden"><a id="L9" href="#L9">     9</a>  </span><span>)</span>
<span class="hidden"><a id="L10" href="#L10">    10</a>  </span>
<span class="hidden"><a id="L11" href="#L11">    11</a>  </span><span>type XO uint8</span>
<span class="hidden"><a id="L12" href="#L12">    12</a>  </span>
<span class="hidden"><a id="L13" href="#L13">    13</a>  </span><span>const (</span>
<span class="hidden"><a id="L14" href="#L14">    14</a>  </span><span>	None XO = iota</span>
<span class="hidden"><a id="L15" href="#L15">    15</a>  </span><span>	X</span>
<span class="hidden"><a id="L16" href="#L16">    16</a>  </span><span>	O</span>
<span class="hidden"><a id="L17" href="#L17">    17</a>  </span><span>)</span>
<span class="hidden"><a id="L18" href="#L18">    18</a>  </span>
<span class="hidden"><a id="L19" href="#L19">    19</a>  </span><span>func (xo XO) Switch() XO {</span>
<span class="hidden"><a id="L20" href="#L20">    20</a>  </span><span>	if xo == None {</span>
<span class="hidden"><a id="L21" href="#L21">    21</a>  </span><span>		return None</span>
<span class="hidden"><a id="L22" href="#L22">    22</a>  </span><span>	}</span>
<span class="hidden"><a id="L23" href="#L23">    23</a>  </span>
<span class="hidden"><a id="L24" href="#L24">    24</a>  </span><span>	return 3 - xo</span>
<span class="hidden"><a id="L25" href="#L25">    25</a>  </span><span>}</span>
<span class="hidden"><a id="L26" href="#L26">    26</a>  </span>
<span class="hidden"><a id="L27" href="#L27">    27</a>  </span><span>func (xo XO) String() string {</span>
<span class="hidden"><a id="L28" href="#L28">    28</a>  </span><span>	switch xo {</span>
<span class="hidden"><a id="L29" href="#L29">    29</a>  </span><span>	case None:</span>
<span class="hidden"><a id="L30" href="#L30">    30</a>  </span><span>		return &#34;None&#34;</span>
<span class="hidden"><a id="L31" href="#L31">    31</a>  </span><span>	case X:</span>
<span class="hidden"><a id="L32" href="#L32">    32</a>  </span><span>		return &#34;X&#34;</span>
<span class="hidden"><a id="L33" href="#L33">    33</a>  </span><span>	case O:</span>
<span class="hidden"><a id="L34" href="#L34">    34</a>  </span><span>		return &#34;O&#34;</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>	return &#34;Invalid&#34;</span>
<span class="hidden"><a id="L38" href="#L38">    38</a>  </span><span>}</span>
<span class="hidden"><a id="L39" href="#L39">    39</a>  </span>
<span class="hidden"><a id="L40" href="#L40">    40</a>  </span><span>type Position uint8</span>
<span class="hidden"><a id="L41" href="#L41">    41</a>  </span>
<span class="hidden"><a id="L42" href="#L42">    42</a>  </span><span>func (p Position) RotateClockwise() Position {</span>
<span class="hidden"><a id="L43" href="#L43">    43</a>  </span><span>	return 2 - p</span><span>/3 &#43; 3*(p%3)</span>
<span class="hidden"><a id="L44" href="#L44">    44</a>  </span><span>}</span>
<span class="hidden"><a id="L45" href="#L45">    45</a>  </span>
<span class="hidden"><a id="L46" href="#L46">    46</a>  </span><span>func (p Position) Flop() Position {</span>
<span class="hidden"><a id="L47" href="#L47">    47</a>  </span><span>	return 3*(p</span><span>/3) &#43; 2 - (p % 3)</span>
<span class="hidden"><a id="L48" href="#L48">    48</a>  </span><span>}</span>
<span class="hidden"><a id="L49" href="#L49">    49</a>  </span>
<span class="hidden"><a id="L50" href="#L50">    50</a>  </span><span>var (</span>
<span class="hidden"><a id="L51" href="#L51">    51</a>  </span><span>	Positions = [...]Position{0, 1, 2, 3, 4, 5, 6, 7, 8}</span>
<span class="hidden"><a id="L52" href="#L52">    52</a>  </span><span>	wins      = [...][3]Position{</span>
<span class="hidden"><a id="L53" href="#L53">    53</a>  </span><span>		{0, 1, 2},</span>
<span class="hidden"><a id="L54" href="#L54">    54</a>  </span><span>		{3, 4, 5},</span>
<span class="hidden"><a id="L55" href="#L55">    55</a>  </span><span>		{6, 7, 8},</span>
<span class="hidden"><a id="L56" href="#L56">    56</a>  </span><span>		{0, 3, 6},</span>
<span class="hidden"><a id="L57" href="#L57">    57</a>  </span><span>		{1, 4, 7},</span>
<span class="hidden"><a id="L58" href="#L58">    58</a>  </span><span>		{2, 5, 8},</span>
<span class="hidden"><a id="L59" href="#L59">    59</a>  </span><span>		{0, 4, 8},</span>
<span class="hidden"><a id="L60" href="#L60">    60</a>  </span><span>		{2, 4, 6},</span>
<span class="hidden"><a id="L61" href="#L61">    61</a>  </span><span>	}</span>
<span class="hidden"><a id="L62" href="#L62">    62</a>  </span><span>)</span>
<span class="hidden"><a id="L63" href="#L63">    63</a>  </span>
<span class="hidden"><a id="L64" href="#L64">    64</a>  </span><span>type Board uint32</span>
<span class="hidden"><a id="L65" href="#L65">    65</a>  </span>
<span class="hidden"><a id="L66" href="#L66">    66</a>  </span><span>const emptyBoard Board = 0</span>
<span class="hidden"><a id="L67" href="#L67">    67</a>  </span>
<span class="hidden"><a id="L68" href="#L68">    68</a>  </span><span>func (b Board) Get(pos Position) XO {</span>
<span class="hidden"><a id="L69" href="#L69">    69</a>  </span><span>	return XO(b&gt;&gt;(pos&lt;&lt;1)) &amp; 3</span>
<span class="hidden"><a id="L70" href="#L70">    70</a>  </span><span>}</span>
<span class="hidden"><a id="L71" href="#L71">    71</a>  </span>
<span class="hidden"><a id="L72" href="#L72">    72</a>  </span><span>func (b Board) Set(pos Position, xo XO) Board {</span>
<span class="hidden"><a id="L73" href="#L73">    73</a>  </span><span>	return (b &amp; ^(3 &lt;&lt; (pos &lt;&lt; 1))) | (Board(xo) &lt;&lt; (pos &lt;&lt; 1))</span>
<span class="hidden"><a id="L74" href="#L74">    74</a>  </span><span>}</span>
<span class="hidden"><a id="L75" href="#L75">    75</a>  </span>
<span class="hidden"><a id="L76" href="#L76">    76</a>  </span><span>func (b Board) Switch() Board {</span>
<span class="hidden"><a id="L77" href="#L77">    77</a>  </span><span>	c := emptyBoard</span>
<span class="hidden"><a id="L78" href="#L78">    78</a>  </span>
<span class="hidden"><a id="L79" href="#L79">    79</a>  </span><span>	for _, p := range Positions {</span>
<span class="hidden"><a id="L80" href="#L80">    80</a>  </span><span>		c = c.Set(p, b.Get(p).Switch())</span>
<span class="hidden"><a id="L81" href="#L81">    81</a>  </span><span>	}</span>
<span class="hidden"><a id="L82" href="#L82">    82</a>  </span>
<span class="hidden"><a id="L83" href="#L83">    83</a>  </span><span>	return c</span>
<span class="hidden"><a id="L84" href="#L84">    84</a>  </span><span>}</span>
<span class="hidden"><a id="L85" href="#L85">    85</a>  </span>
<span class="hidden"><a id="L86" href="#L86">    86</a>  </span><span>func (b Board) Transform(flop bool, rotate uint8) Board {</span>
<span class="hidden"><a id="L87" href="#L87">    87</a>  </span><span>	c := emptyBoard</span>
<span class="hidden"><a id="L88" href="#L88">    88</a>  </span>
<span class="hidden"><a id="L89" href="#L89">    89</a>  </span><span>	for _, p := range Positions {</span>
<span class="hidden"><a id="L90" href="#L90">    90</a>  </span><span>		v := b.Get(p)</span>
<span class="hidden"><a id="L91" href="#L91">    91</a>  </span>
<span class="hidden"><a id="L92" href="#L92">    92</a>  </span><span>		if flop {</span>
<span class="hidden"><a id="L93" href="#L93">    93</a>  </span><span>			p = p.Flop()</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>		for r := uint8(0); r &lt; rotate; r&#43;&#43; {</span>
<span class="hidden"><a id="L97" href="#L97">    97</a>  </span><span>			p = p.RotateClockwise()</span>
<span class="hidden"><a id="L98" href="#L98">    98</a>  </span><span>		}</span>
<span class="hidden"><a id="L99" href="#L99">    99</a>  </span>
<span class="hidden"><a id="L100" href="#L100">   100</a>  </span><span>		c = c.Set(p, v)</span>
<span class="hidden"><a id="L101" href="#L101">   101</a>  </span><span>	}</span>
<span class="hidden"><a id="L102" href="#L102">   102</a>  </span>
<span class="hidden"><a id="L103" href="#L103">   103</a>  </span><span>	return c</span>
<span class="hidden"><a id="L104" href="#L104">   104</a>  </span><span>}</span>
<span class="hidden"><a id="L105" href="#L105">   105</a>  </span>
<span class="hidden"><a id="L106" href="#L106">   106</a>  </span><span>func (b Board) HasWin() bool {</span>
<span class="hidden"><a id="L107" href="#L107">   107</a>  </span><span>	return b.hasXsInARow(3)</span>
<span class="hidden"><a id="L108" href="#L108">   108</a>  </span><span>}</span>
<span class="hidden"><a id="L109" href="#L109">   109</a>  </span>
<span class="hidden"><a id="L110" href="#L110">   110</a>  </span><span>func (b Board) hasXsInARow(num int) bool {</span>
<span class="hidden"><a id="L111" href="#L111">   111</a>  </span><span>Loop:</span>
<span class="hidden"><a id="L112" href="#L112">   112</a>  </span><span>	for _, w := range wins {</span>
<span class="hidden"><a id="L113" href="#L113">   113</a>  </span><span>		xs := 0</span>
<span class="hidden"><a id="L114" href="#L114">   114</a>  </span>
<span class="hidden"><a id="L115" href="#L115">   115</a>  </span><span>		for _, wn := range w {</span>
<span class="hidden"><a id="L116" href="#L116">   116</a>  </span><span>			switch b.Get(wn) {</span>
<span class="hidden"><a id="L117" href="#L117">   117</a>  </span><span>			case X:</span>
<span class="hidden"><a id="L118" href="#L118">   118</a>  </span><span>				xs&#43;&#43;</span>
<span class="hidden"><a id="L119" href="#L119">   119</a>  </span><span>			case O:</span>
<span class="hidden"><a id="L120" href="#L120">   120</a>  </span><span>				continue Loop</span>
<span class="hidden"><a id="L121" href="#L121">   121</a>  </span><span>			}</span>
<span class="hidden"><a id="L122" href="#L122">   122</a>  </span><span>		}</span>
<span class="hidden"><a id="L123" href="#L123">   123</a>  </span>
<span class="hidden"><a id="L124" href="#L124">   124</a>  </span><span>		if xs == num {</span>
<span class="hidden"><a id="L125" href="#L125">   125</a>  </span><span>			return true</span>
<span class="hidden"><a id="L126" href="#L126">   126</a>  </span><span>		}</span>
<span class="hidden"><a id="L127" href="#L127">   127</a>  </span><span>	}</span>
<span class="hidden"><a id="L128" href="#L128">   128</a>  </span>
<span class="hidden"><a id="L129" href="#L129">   129</a>  </span><span>	return false</span>
<span class="hidden"><a id="L130" href="#L130">   130</a>  </span><span>}</span>
<span class="hidden"><a id="L131" href="#L131">   131</a>  </span>
<span class="hidden"><a id="L132" href="#L132">   132</a>  </span><span>func (b Board) HasPossibleWin() bool {</span>
<span class="hidden"><a id="L133" href="#L133">   133</a>  </span><span>	return b.hasXsInARow(2)</span>
<span class="hidden"><a id="L134" href="#L134">   134</a>  </span><span>}</span>
<span class="hidden"><a id="L135" href="#L135">   135</a>  </span>
<span class="hidden"><a id="L136" href="#L136">   136</a>  </span><span>func (b Board) String() string {</span>
<span class="hidden"><a id="L137" href="#L137">   137</a>  </span><span>	var sb strings.Builder</span>
<span class="hidden"><a id="L138" href="#L138">   138</a>  </span>
<span class="hidden"><a id="L139" href="#L139">   139</a>  </span><span>	sb.WriteString(&#34;┌───┬───┬───┐\n&#34;)</span>
<span class="hidden"><a id="L140" href="#L140">   140</a>  </span><span>	for y := Position(0); y &lt; 3; y&#43;&#43; {</span>
<span class="hidden"><a id="L141" href="#L141">   141</a>  </span><span>		if y &gt; 0 {</span>
<span class="hidden"><a id="L142" href="#L142">   142</a>  </span><span>			sb.WriteString(&#34;├───┼───┼───┤\n&#34;)</span>
<span class="hidden"><a id="L143" href="#L143">   143</a>  </span><span>		}</span>
<span class="hidden"><a id="L144" href="#L144">   144</a>  </span>
<span class="hidden"><a id="L145" href="#L145">   145</a>  </span><span>		for x := Position(0); x &lt; 3; x&#43;&#43; {</span>
<span class="hidden"><a id="L146" href="#L146">   146</a>  </span><span>			sb.WriteString(&#34;│ &#34;)</span>
<span class="hidden"><a id="L147" href="#L147">   147</a>  </span><span>			if p := b.Get(y*3 &#43; x); p == None {</span>
<span class="hidden"><a id="L148" href="#L148">   148</a>  </span><span>				sb.WriteString(&#34; &#34;)</span>
<span class="hidden"><a id="L149" href="#L149">   149</a>  </span><span>			} else {</span>
<span class="hidden"><a id="L150" href="#L150">   150</a>  </span><span>				sb.WriteString(p.String())</span>
<span class="hidden"><a id="L151" href="#L151">   151</a>  </span><span>			}</span>
<span class="hidden"><a id="L152" href="#L152">   152</a>  </span><span>			sb.WriteString(&#34; &#34;)</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 class="hidden"><a id="L155" href="#L155">   155</a>  </span><span>		sb.WriteString(&#34;│\n&#34;)</span>
<span class="hidden"><a id="L156" href="#L156">   156</a>  </span><span>	}</span>
<span class="hidden"><a id="L157" href="#L157">   157</a>  </span><span>	sb.WriteString(&#34;└───┴───┴───┘&#34;)</span>
<span class="hidden"><a id="L158" href="#L158">   158</a>  </span>
<span class="hidden"><a id="L159" href="#L159">   159</a>  </span><span>	return sb.String()</span>
<span class="hidden"><a id="L160" href="#L160">   160</a>  </span><span>}</span>
<span class="hidden"><a id="L161" href="#L161">   161</a>  </span>
<span class="hidden"><a id="L162" href="#L162">   162</a>  </span><span>type Result uint8</span>
<span class="hidden"><a id="L163" href="#L163">   163</a>  </span>
<span class="hidden"><a id="L164" href="#L164">   164</a>  </span><span>const (</span>
<span class="hidden"><a id="L165" href="#L165">   165</a>  </span><span>	WillLose Result = iota</span>
<span class="hidden"><a id="L166" href="#L166">   166</a>  </span><span>	CanLose</span>
<span class="hidden"><a id="L167" href="#L167">   167</a>  </span><span>	Draw</span>
<span class="hidden"><a id="L168" href="#L168">   168</a>  </span><span>	CanWin</span>
<span class="hidden"><a id="L169" href="#L169">   169</a>  </span><span>	WillWin</span>
<span class="hidden"><a id="L170" href="#L170">   170</a>  </span><span>	FilledX</span>
<span class="hidden"><a id="L171" href="#L171">   171</a>  </span><span>	FilledO</span>
<span class="hidden"><a id="L172" href="#L172">   172</a>  </span><span>)</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 (r Result) String() string {</span>
<span class="hidden"><a id="L175" href="#L175">   175</a>  </span><span>	switch r {</span>
<span class="hidden"><a id="L176" href="#L176">   176</a>  </span><span>	case WillLose:</span>
<span class="hidden"><a id="L177" href="#L177">   177</a>  </span><span>		return &#34;Will Lose&#34;</span>
<span class="hidden"><a id="L178" href="#L178">   178</a>  </span><span>	case CanLose:</span>
<span class="hidden"><a id="L179" href="#L179">   179</a>  </span><span>		return &#34;Can Lose&#34;</span>
<span class="hidden"><a id="L180" href="#L180">   180</a>  </span><span>	case Draw:</span>
<span class="hidden"><a id="L181" href="#L181">   181</a>  </span><span>		return &#34;Draw&#34;</span>
<span class="hidden"><a id="L182" href="#L182">   182</a>  </span><span>	case CanWin:</span>
<span class="hidden"><a id="L183" href="#L183">   183</a>  </span><span>		return &#34;Can Win&#34;</span>
<span class="hidden"><a id="L184" href="#L184">   184</a>  </span><span>	case WillWin:</span>
<span class="hidden"><a id="L185" href="#L185">   185</a>  </span><span>		return &#34;Will Win&#34;</span>
<span class="hidden"><a id="L186" href="#L186">   186</a>  </span><span>	case FilledX:</span>
<span class="hidden"><a id="L187" href="#L187">   187</a>  </span><span>		return &#34;Filled X&#34;</span>
<span class="hidden"><a id="L188" href="#L188">   188</a>  </span><span>	case FilledO:</span>
<span class="hidden"><a id="L189" href="#L189">   189</a>  </span><span>		return &#34;Filled O&#34;</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>	return &#34;Invalid&#34;</span>
<span class="hidden"><a id="L193" href="#L193">   193</a>  </span><span>}</span>
<span class="hidden"><a id="L194" href="#L194">   194</a>  </span>
<span class="hidden"><a id="L195" href="#L195">   195</a>  </span><span>func (r Result) Switch() Result {</span>
<span class="hidden"><a id="L196" href="#L196">   196</a>  </span><span>	if r &lt;= WillWin {</span>
<span class="hidden"><a id="L197" href="#L197">   197</a>  </span><span>		return WillWin - r</span>
<span class="hidden"><a id="L198" href="#L198">   198</a>  </span><span>	}</span>
<span class="hidden"><a id="L199" href="#L199">   199</a>  </span>
<span class="hidden"><a id="L200" href="#L200">   200</a>  </span><span>	return r</span>
<span class="hidden"><a id="L201" href="#L201">   201</a>  </span><span>}</span>
<span class="hidden"><a id="L202" href="#L202">   202</a>  </span>
<span class="hidden"><a id="L203" href="#L203">   203</a>  </span><span>type Results uint32</span>
<span class="hidden"><a id="L204" href="#L204">   204</a>  </span>
<span class="hidden"><a id="L205" href="#L205">   205</a>  </span><span>func (rs Results) Get(p Position) Result {</span>
<span class="hidden"><a id="L206" href="#L206">   206</a>  </span><span>	return Result((rs &gt;&gt; (p * 3)) &amp; 7)</span>
<span class="hidden"><a id="L207" href="#L207">   207</a>  </span><span>}</span>
<span class="hidden"><a id="L208" href="#L208">   208</a>  </span>
<span class="hidden"><a id="L209" href="#L209">   209</a>  </span><span>func (rs Results) Set(p Position, r Result) Results {</span>
<span class="hidden"><a id="L210" href="#L210">   210</a>  </span><span>	return (rs &amp; ^(7 &lt;&lt; (p * 3))) | (Results(r) &lt;&lt; (p * 3))</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 class="hidden"><a id="L213" href="#L213">   213</a>  </span><span>func (rs Results) SetState(b Result) Results {</span>
<span class="hidden"><a id="L214" href="#L214">   214</a>  </span><span>	return rs | (Results(b) &lt;&lt; 27)</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>func (rs Results) GetState() Result {</span>
<span class="hidden"><a id="L218" href="#L218">   218</a>  </span><span>	return Result(rs &gt;&gt; 27)</span>
<span class="hidden"><a id="L219" href="#L219">   219</a>  </span><span>}</span>
<span class="hidden"><a id="L220" href="#L220">   220</a>  </span>
<span class="hidden"><a id="L221" href="#L221">   221</a>  </span><span>func (rs Results) String() string {</span>
<span class="hidden"><a id="L222" href="#L222">   222</a>  </span><span>	var sb strings.Builder</span>
<span class="hidden"><a id="L223" href="#L223">   223</a>  </span>
<span class="hidden"><a id="L224" href="#L224">   224</a>  </span><span>	sb.WriteString(&#34;Overall: &#34;)</span>
<span class="hidden"><a id="L225" href="#L225">   225</a>  </span><span>	sb.WriteString(rs.GetState().String())</span>
<span class="hidden"><a id="L226" href="#L226">   226</a>  </span><span>	sb.WriteString(&#34;\n┌───┬───┬───┐\n&#34;)</span>
<span class="hidden"><a id="L227" href="#L227">   227</a>  </span><span>	for y := Position(0); y &lt; 3; y&#43;&#43; {</span>
<span class="hidden"><a id="L228" href="#L228">   228</a>  </span><span>		if y &gt; 0 {</span>
<span class="hidden"><a id="L229" href="#L229">   229</a>  </span><span>			sb.WriteString(&#34;├───┼───┼───┤\n&#34;)</span>
<span class="hidden"><a id="L230" href="#L230">   230</a>  </span><span>		}</span>
<span class="hidden"><a id="L231" href="#L231">   231</a>  </span>
<span class="hidden"><a id="L232" href="#L232">   232</a>  </span><span>		for x := Position(0); x &lt; 3; x&#43;&#43; {</span>
<span class="hidden"><a id="L233" href="#L233">   233</a>  </span><span>			sb.WriteString(&#34;│ &#34;)</span>
<span class="hidden"><a id="L234" href="#L234">   234</a>  </span><span>			switch rs.Get(y*3 &#43; x) {</span>
<span class="hidden"><a id="L235" href="#L235">   235</a>  </span><span>			case WillLose:</span>
<span class="hidden"><a id="L236" href="#L236">   236</a>  </span><span>				sb.WriteString(&#34;L&#34;)</span>
<span class="hidden"><a id="L237" href="#L237">   237</a>  </span><span>			case CanLose:</span>
<span class="hidden"><a id="L238" href="#L238">   238</a>  </span><span>				sb.WriteString(&#34;l&#34;)</span>
<span class="hidden"><a id="L239" href="#L239">   239</a>  </span><span>			case Draw:</span>
<span class="hidden"><a id="L240" href="#L240">   240</a>  </span><span>				sb.WriteString(&#34;D&#34;)</span>
<span class="hidden"><a id="L241" href="#L241">   241</a>  </span><span>			case CanWin:</span>
<span class="hidden"><a id="L242" href="#L242">   242</a>  </span><span>				sb.WriteString(&#34;w&#34;)</span>
<span class="hidden"><a id="L243" href="#L243">   243</a>  </span><span>			case WillWin:</span>
<span class="hidden"><a id="L244" href="#L244">   244</a>  </span><span>				sb.WriteString(&#34;W&#34;)</span>
<span class="hidden"><a id="L245" href="#L245">   245</a>  </span><span>			case FilledX:</span>
<span class="hidden"><a id="L246" href="#L246">   246</a>  </span><span>				sb.WriteString(&#34;X&#34;)</span>
<span class="hidden"><a id="L247" href="#L247">   247</a>  </span><span>			case FilledO:</span>
<span class="hidden"><a id="L248" href="#L248">   248</a>  </span><span>				sb.WriteString(&#34;O&#34;)</span>
<span class="hidden"><a id="L249" href="#L249">   249</a>  </span><span>			}</span>
<span class="hidden"><a id="L250" href="#L250">   250</a>  </span><span>			sb.WriteString(&#34; &#34;)</span>
<span class="hidden"><a id="L251" href="#L251">   251</a>  </span><span>		}</span>
<span class="hidden"><a id="L252" href="#L252">   252</a>  </span>
<span class="hidden"><a id="L253" href="#L253">   253</a>  </span><span>		sb.WriteString(&#34;│\n&#34;)</span>
<span class="hidden"><a id="L254" href="#L254">   254</a>  </span><span>	}</span>
<span class="hidden"><a id="L255" href="#L255">   255</a>  </span><span>	sb.WriteString(&#34;└───┴───┴───┘&#34;)</span>
<span class="hidden"><a id="L256" href="#L256">   256</a>  </span>
<span class="hidden"><a id="L257" href="#L257">   257</a>  </span><span>	return sb.String()</span>
<span class="hidden"><a id="L258" href="#L258">   258</a>  </span><span>}</span>
<span class="hidden"><a id="L259" href="#L259">   259</a>  </span>
<span class="hidden"><a id="L260" href="#L260">   260</a>  </span><span>func (r Results) Encode() [3]byte {</span>
<span class="hidden"><a id="L261" href="#L261">   261</a>  </span><span>	var n uint32</span>
<span class="hidden"><a id="L262" href="#L262">   262</a>  </span>
<span class="hidden"><a id="L263" href="#L263">   263</a>  </span><span>	pow := uint32(1)</span>
<span class="hidden"><a id="L264" href="#L264">   264</a>  </span>
<span class="hidden"><a id="L265" href="#L265">   265</a>  </span><span>	for _, p := range Positions {</span>
<span class="hidden"><a id="L266" href="#L266">   266</a>  </span><span>		v := r.Get(p)</span>
<span class="hidden"><a id="L267" href="#L267">   267</a>  </span>
<span class="hidden"><a id="L268" href="#L268">   268</a>  </span><span>		if v &gt; CanLose {</span>
<span class="hidden"><a id="L269" href="#L269">   269</a>  </span><span>			v--</span>
<span class="hidden"><a id="L270" href="#L270">   270</a>  </span><span>		}</span>
<span class="hidden"><a id="L271" href="#L271">   271</a>  </span>
<span class="hidden"><a id="L272" href="#L272">   272</a>  </span><span>		n &#43;= uint32(v) * pow</span>
<span class="hidden"><a id="L273" href="#L273">   273</a>  </span>
<span class="hidden"><a id="L274" href="#L274">   274</a>  </span><span>		pow *= 6</span>
<span class="hidden"><a id="L275" href="#L275">   275</a>  </span><span>	}</span>
<span class="hidden"><a id="L276" href="#L276">   276</a>  </span>
<span class="hidden"><a id="L277" href="#L277">   277</a>  </span><span>	return [3]byte{</span>
<span class="hidden"><a id="L278" href="#L278">   278</a>  </span><span>		byte(n &gt;&gt; 16),</span>
<span class="hidden"><a id="L279" href="#L279">   279</a>  </span><span>		byte(n &gt;&gt; 8),</span>
<span class="hidden"><a id="L280" href="#L280">   280</a>  </span><span>		byte(n),</span>
<span class="hidden"><a id="L281" href="#L281">   281</a>  </span><span>	}</span>
<span class="hidden"><a id="L282" href="#L282">   282</a>  </span><span>}</span>
<span class="hidden"><a id="L283" href="#L283">   283</a>  </span>
<span class="hidden"><a id="L284" href="#L284">   284</a>  </span><span>type Brain map[Board]Results</span>
<span class="hidden"><a id="L285" href="#L285">   285</a>  </span>
<span class="hidden"><a id="L286" href="#L286">   286</a>  </span><span>func NewBrain() Brain {</span>
<span class="hidden"><a id="L287" href="#L287">   287</a>  </span><span>	b := make(Brain)</span>
<span class="hidden"><a id="L288" href="#L288">   288</a>  </span>
<span class="hidden"><a id="L289" href="#L289">   289</a>  </span><span>	b.move(emptyBoard)</span>
<span class="hidden"><a id="L290" href="#L290">   290</a>  </span>
<span class="hidden"><a id="L291" href="#L291">   291</a>  </span><span>	return b</span>
<span class="hidden"><a id="L292" href="#L292">   292</a>  </span><span>}</span>
<span class="hidden"><a id="L293" href="#L293">   293</a>  </span>
<span class="hidden"><a id="L294" href="#L294">   294</a>  </span><span>func (b Brain) move(board Board) Result {</span>
<span class="hidden"><a id="L295" href="#L295">   295</a>  </span><span>	if r, ok := b.getResults(board); ok {</span>
<span class="hidden"><a id="L296" href="#L296">   296</a>  </span><span>		return r.GetState()</span>
<span class="hidden"><a id="L297" href="#L297">   297</a>  </span><span>	}</span>
<span class="hidden"><a id="L298" href="#L298">   298</a>  </span>
<span class="hidden"><a id="L299" href="#L299">   299</a>  </span><span>	var (</span>
<span class="hidden"><a id="L300" href="#L300">   300</a>  </span><span>		rs       Results</span>
<span class="hidden"><a id="L301" href="#L301">   301</a>  </span><span>		result   Result</span>
<span class="hidden"><a id="L302" href="#L302">   302</a>  </span><span>		hasEmpty bool</span>
<span class="hidden"><a id="L303" href="#L303">   303</a>  </span><span>	)</span>
<span class="hidden"><a id="L304" href="#L304">   304</a>  </span>
<span class="hidden"><a id="L305" href="#L305">   305</a>  </span><span>	for _, p := range Positions {</span>
<span class="hidden"><a id="L306" href="#L306">   306</a>  </span><span>		if t := board.Get(p); t != None {</span>
<span class="hidden"><a id="L307" href="#L307">   307</a>  </span><span>			rs = rs.Set(p, FilledX&#43;Result(t-1))</span>
<span class="hidden"><a id="L308" href="#L308">   308</a>  </span><span>			continue</span>
<span class="hidden"><a id="L309" href="#L309">   309</a>  </span><span>		}</span>
<span class="hidden"><a id="L310" href="#L310">   310</a>  </span>
<span class="hidden"><a id="L311" href="#L311">   311</a>  </span><span>		hasEmpty = true</span>
<span class="hidden"><a id="L312" href="#L312">   312</a>  </span>
<span class="hidden"><a id="L313" href="#L313">   313</a>  </span><span>		setBoard := board.Set(p, X)</span>
<span class="hidden"><a id="L314" href="#L314">   314</a>  </span>
<span class="hidden"><a id="L315" href="#L315">   315</a>  </span><span>		var r Result</span>
<span class="hidden"><a id="L316" href="#L316">   316</a>  </span>
<span class="hidden"><a id="L317" href="#L317">   317</a>  </span><span>		if setBoard.HasWin() {</span>
<span class="hidden"><a id="L318" href="#L318">   318</a>  </span><span>			r = WillWin</span>
<span class="hidden"><a id="L319" href="#L319">   319</a>  </span><span>		} else {</span>
<span class="hidden"><a id="L320" href="#L320">   320</a>  </span><span>			r = b.move(setBoard.Switch()).Switch()</span>
<span class="hidden"><a id="L321" href="#L321">   321</a>  </span>
<span class="hidden"><a id="L322" href="#L322">   322</a>  </span><span>			if (r == Draw || r == CanLose) &amp;&amp; setBoard.HasPossibleWin() {</span>
<span class="hidden"><a id="L323" href="#L323">   323</a>  </span><span>				r = CanWin</span>
<span class="hidden"><a id="L324" href="#L324">   324</a>  </span><span>			}</span>
<span class="hidden"><a id="L325" href="#L325">   325</a>  </span><span>		}</span>
<span class="hidden"><a id="L326" href="#L326">   326</a>  </span>
<span class="hidden"><a id="L327" href="#L327">   327</a>  </span><span>		if r &gt; result {</span>
<span class="hidden"><a id="L328" href="#L328">   328</a>  </span><span>			result = r</span>
<span class="hidden"><a id="L329" href="#L329">   329</a>  </span><span>		}</span>
<span class="hidden"><a id="L330" href="#L330">   330</a>  </span>
<span class="hidden"><a id="L331" href="#L331">   331</a>  </span><span>		rs = rs.Set(p, r)</span>
<span class="hidden"><a id="L332" href="#L332">   332</a>  </span><span>	}</span>
<span class="hidden"><a id="L333" href="#L333">   333</a>  </span>
<span class="hidden"><a id="L334" href="#L334">   334</a>  </span><span>	if !hasEmpty {</span>
<span class="hidden"><a id="L335" href="#L335">   335</a>  </span><span>		result = Draw</span>
<span class="hidden"><a id="L336" href="#L336">   336</a>  </span><span>	}</span>
<span class="hidden"><a id="L337" href="#L337">   337</a>  </span>
<span class="hidden"><a id="L338" href="#L338">   338</a>  </span><span>	b[board] = rs.SetState(result)</span>
<span class="hidden"><a id="L339" href="#L339">   339</a>  </span>
<span class="hidden"><a id="L340" href="#L340">   340</a>  </span><span>	return result</span>
<span class="hidden"><a id="L341" href="#L341">   341</a>  </span><span>}</span>
<span class="hidden"><a id="L342" href="#L342">   342</a>  </span>
<span class="hidden"><a id="L343" href="#L343">   343</a>  </span><span>func (b Brain) getResults(board Board) (Results, bool) {</span>
<span class="hidden"><a id="L344" href="#L344">   344</a>  </span><span>	for i := uint8(0); i &lt; 8; i&#43;&#43; {</span>
<span class="hidden"><a id="L345" href="#L345">   345</a>  </span><span>		if r, ok := b[board.Transform(i&amp;4 != 0, i&amp;3)]; ok {</span>
<span class="hidden"><a id="L346" href="#L346">   346</a>  </span><span>			return r, true</span>
<span class="hidden"><a id="L347" href="#L347">   347</a>  </span><span>		}</span>
<span class="hidden"><a id="L348" href="#L348">   348</a>  </span><span>	}</span>
<span class="hidden"><a id="L349" href="#L349">   349</a>  </span>
<span class="hidden"><a id="L350" href="#L350">   350</a>  </span><span>	return 0, false</span>
<span class="hidden"><a id="L351" href="#L351">   351</a>  </span><span>}</span>
<span class="hidden"><a id="L352" href="#L352">   352</a>  </span>
<span class="hidden"><a id="L353" href="#L353">   353</a>  </span><span>func main() {</span>
<span class="hidden"><a id="L354" href="#L354">   354</a>  </span><span>	var debug, js bool</span>
<span class="hidden"><a id="L355" href="#L355">   355</a>  </span>
<span class="hidden"><a id="L356" href="#L356">   356</a>  </span><span>	flag.BoolVar(&amp;debug, &#34;d&#34;, false, &#34;print all boards&#34;)</span>
<span class="hidden"><a id="L357" href="#L357">   357</a>  </span><span>	flag.BoolVar(&amp;js, &#34;j&#34;, false, &#34;print brain wrapped in JS&#34;)</span>
<span class="hidden"><a id="L358" href="#L358">   358</a>  </span>
<span class="hidden"><a id="L359" href="#L359">   359</a>  </span><span>	flag.Parse()</span>
<span class="hidden"><a id="L360" href="#L360">   360</a>  </span>
<span class="hidden"><a id="L361" href="#L361">   361</a>  </span><span>	b := NewBrain()</span>
<span class="hidden"><a id="L362" href="#L362">   362</a>  </span>
<span class="hidden"><a id="L363" href="#L363">   363</a>  </span><span>	boards := make([]Board, 0, len(b))</span>
<span class="hidden"><a id="L364" href="#L364">   364</a>  </span>
<span class="hidden"><a id="L365" href="#L365">   365</a>  </span><span>	for board := range b {</span>
<span class="hidden"><a id="L366" href="#L366">   366</a>  </span><span>		boards = append(boards, board)</span>
<span class="hidden"><a id="L367" href="#L367">   367</a>  </span><span>	}</span>
<span class="hidden"><a id="L368" href="#L368">   368</a>  </span>
<span class="hidden"><a id="L369" href="#L369">   369</a>  </span><span>	slices.Sort(boards)</span>
<span class="hidden"><a id="L370" href="#L370">   370</a>  </span>
<span class="hidden"><a id="L371" href="#L371">   371</a>  </span><span>	if debug {</span>
<span class="hidden"><a id="L372" href="#L372">   372</a>  </span><span>		for _, board := range boards {</span>
<span class="hidden"><a id="L373" href="#L373">   373</a>  </span><span>			results := b[board]</span>
<span class="hidden"><a id="L374" href="#L374">   374</a>  </span><span>			b := strings.Split(board.String(), &#34;\n&#34;)</span>
<span class="hidden"><a id="L375" href="#L375">   375</a>  </span><span>			r := strings.Split(results.String(), &#34;\n&#34;)</span>
<span class="hidden"><a id="L376" href="#L376">   376</a>  </span>
<span class="hidden"><a id="L377" href="#L377">   377</a>  </span><span>			fmt.Println(&#34;             &#34;, r[0])</span>
<span class="hidden"><a id="L378" href="#L378">   378</a>  </span>
<span class="hidden"><a id="L379" href="#L379">   379</a>  </span><span>			for n, rs := range r[1:] {</span>
<span class="hidden"><a id="L380" href="#L380">   380</a>  </span><span>				fmt.Println(b[n], rs)</span>
<span class="hidden"><a id="L381" href="#L381">   381</a>  </span><span>			}</span>
<span class="hidden"><a id="L382" href="#L382">   382</a>  </span>
<span class="hidden"><a id="L383" href="#L383">   383</a>  </span><span>		}</span>
<span class="hidden"><a id="L384" href="#L384">   384</a>  </span>
<span class="hidden"><a id="L385" href="#L385">   385</a>  </span><span>		fmt.Printf(&#34;%d boards\n&#34;, len(b))</span>
<span class="hidden"><a id="L386" href="#L386">   386</a>  </span><span>	} else {</span>
<span class="hidden"><a id="L387" href="#L387">   387</a>  </span><span>		var sb strings.Builder</span>
<span class="hidden"><a id="L388" href="#L388">   388</a>  </span>
<span class="hidden"><a id="L389" href="#L389">   389</a>  </span><span>		w := base64.NewEncoder(base64.StdEncoding, &amp;sb)</span>
<span class="hidden"><a id="L390" href="#L390">   390</a>  </span>
<span class="hidden"><a id="L391" href="#L391">   391</a>  </span><span>		for _, board := range boards {</span>
<span class="hidden"><a id="L392" href="#L392">   392</a>  </span><span>			v := b[board].Encode()</span>
<span class="hidden"><a id="L393" href="#L393">   393</a>  </span>
<span class="hidden"><a id="L394" href="#L394">   394</a>  </span><span>			w.Write(v[:])</span>
<span class="hidden"><a id="L395" href="#L395">   395</a>  </span><span>		}</span>
<span class="hidden"><a id="L396" href="#L396">   396</a>  </span>
<span class="hidden"><a id="L397" href="#L397">   397</a>  </span><span>		w.Close()</span>
<span class="hidden"><a id="L398" href="#L398">   398</a>  </span>
<span class="hidden"><a id="L399" href="#L399">   399</a>  </span><span>		if js {</span>
<span class="hidden"><a id="L400" href="#L400">   400</a>  </span><span>			fmt.Printf(&#34;export default %q;&#34;, sb.String())</span>
<span class="hidden"><a id="L401" href="#L401">   401</a>  </span><span>		} else {</span>
<span class="hidden"><a id="L402" href="#L402">   402</a>  </span><span>			fmt.Println(sb.String())</span>
<span class="hidden"><a id="L403" href="#L403">   403</a>  </span><span>		}</span>
<span class="hidden"><a id="L404" href="#L404">   404</a>  </span><span>	}</span>
<span class="hidden"><a id="L405" href="#L405">   405</a>  </span><span>}</span>
<span class="hidden"><a id="L406" href="#L406">   406</a>  </span></pre>
	</body>
</html>
