ronin-support 0.4.1 → 0.5.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (224) hide show
  1. checksums.yaml +7 -0
  2. data/.editorconfig +11 -0
  3. data/.github/workflows/ruby.yml +28 -0
  4. data/.ruby-version +1 -0
  5. data/.yardopts +1 -1
  6. data/ChangeLog.md +121 -33
  7. data/Gemfile +17 -18
  8. data/README.md +57 -33
  9. data/Rakefile +10 -3
  10. data/gemspec.yml +16 -7
  11. data/lib/ronin/binary.rb +21 -0
  12. data/lib/ronin/binary/hexdump.rb +20 -0
  13. data/lib/ronin/binary/hexdump/parser.rb +403 -0
  14. data/lib/ronin/binary/struct.rb +567 -0
  15. data/lib/ronin/binary/template.rb +454 -0
  16. data/lib/ronin/extensions.rb +5 -5
  17. data/lib/ronin/extensions/enumerable.rb +5 -5
  18. data/lib/ronin/extensions/file.rb +39 -33
  19. data/lib/ronin/extensions/ip_addr.rb +29 -31
  20. data/lib/ronin/extensions/kernel.rb +5 -5
  21. data/lib/ronin/extensions/meta.rb +5 -5
  22. data/lib/ronin/extensions/regexp.rb +50 -5
  23. data/lib/ronin/extensions/resolv.rb +7 -9
  24. data/lib/ronin/extensions/string.rb +10 -10
  25. data/lib/ronin/formatting.rb +5 -5
  26. data/lib/ronin/formatting/binary.rb +5 -5
  27. data/lib/ronin/formatting/digest.rb +5 -5
  28. data/lib/ronin/formatting/extensions.rb +5 -5
  29. data/lib/ronin/formatting/extensions/binary.rb +7 -5
  30. data/lib/ronin/formatting/extensions/binary/array.rb +61 -0
  31. data/lib/ronin/formatting/extensions/binary/base64.rb +106 -0
  32. data/lib/ronin/formatting/extensions/binary/file.rb +44 -11
  33. data/lib/ronin/formatting/extensions/binary/float.rb +65 -0
  34. data/lib/ronin/formatting/extensions/binary/integer.rb +66 -50
  35. data/lib/ronin/formatting/extensions/binary/string.rb +81 -205
  36. data/lib/ronin/formatting/extensions/digest.rb +5 -5
  37. data/lib/ronin/formatting/extensions/digest/file.rb +5 -5
  38. data/lib/ronin/formatting/extensions/digest/string.rb +5 -5
  39. data/lib/ronin/formatting/extensions/html.rb +5 -5
  40. data/lib/ronin/formatting/extensions/html/integer.rb +9 -13
  41. data/lib/ronin/formatting/extensions/html/string.rb +31 -39
  42. data/lib/ronin/formatting/extensions/http.rb +5 -5
  43. data/lib/ronin/formatting/extensions/http/integer.rb +6 -6
  44. data/lib/ronin/formatting/extensions/http/string.rb +7 -7
  45. data/lib/ronin/formatting/extensions/sql.rb +5 -5
  46. data/lib/ronin/formatting/extensions/sql/string.rb +22 -24
  47. data/lib/ronin/formatting/extensions/text.rb +5 -5
  48. data/lib/ronin/formatting/extensions/text/array.rb +13 -11
  49. data/lib/ronin/formatting/extensions/text/string.rb +70 -13
  50. data/lib/ronin/formatting/html.rb +5 -5
  51. data/lib/ronin/formatting/http.rb +5 -5
  52. data/lib/ronin/formatting/sql.rb +5 -5
  53. data/lib/ronin/formatting/text.rb +5 -5
  54. data/lib/ronin/fuzzing.rb +5 -5
  55. data/lib/ronin/fuzzing/extensions.rb +5 -5
  56. data/lib/ronin/fuzzing/extensions/string.rb +42 -213
  57. data/lib/ronin/fuzzing/fuzzer.rb +110 -0
  58. data/lib/ronin/fuzzing/fuzzing.rb +33 -26
  59. data/lib/ronin/fuzzing/mutator.rb +161 -0
  60. data/lib/ronin/fuzzing/repeater.rb +81 -0
  61. data/lib/ronin/fuzzing/template.rb +133 -0
  62. data/lib/ronin/mixin.rb +2 -2
  63. data/lib/ronin/network.rb +7 -5
  64. data/lib/ronin/network/dns.rb +64 -24
  65. data/lib/ronin/network/esmtp.rb +5 -5
  66. data/lib/ronin/network/extensions.rb +5 -5
  67. data/lib/ronin/network/extensions/dns.rb +5 -5
  68. data/lib/ronin/network/extensions/dns/net.rb +5 -5
  69. data/lib/ronin/network/extensions/esmtp.rb +5 -5
  70. data/lib/ronin/network/extensions/esmtp/net.rb +5 -5
  71. data/lib/ronin/network/extensions/http.rb +5 -5
  72. data/lib/ronin/network/extensions/http/net.rb +5 -5
  73. data/lib/ronin/network/extensions/http/uri/http.rb +5 -5
  74. data/lib/ronin/network/extensions/imap.rb +5 -5
  75. data/lib/ronin/network/extensions/imap/net.rb +5 -5
  76. data/lib/ronin/network/extensions/pop3.rb +5 -5
  77. data/lib/ronin/network/extensions/pop3/net.rb +5 -5
  78. data/lib/ronin/network/extensions/smtp.rb +5 -5
  79. data/lib/ronin/network/extensions/smtp/net.rb +5 -5
  80. data/lib/ronin/network/extensions/ssl.rb +5 -5
  81. data/lib/ronin/network/extensions/ssl/net.rb +5 -5
  82. data/lib/ronin/network/extensions/tcp.rb +5 -5
  83. data/lib/ronin/network/extensions/tcp/net.rb +5 -5
  84. data/lib/ronin/network/extensions/telnet.rb +5 -5
  85. data/lib/ronin/network/extensions/telnet/net.rb +5 -5
  86. data/lib/ronin/network/extensions/udp.rb +5 -5
  87. data/lib/ronin/network/extensions/udp/net.rb +5 -5
  88. data/lib/ronin/network/ftp.rb +149 -0
  89. data/lib/ronin/network/http.rb +5 -5
  90. data/lib/ronin/network/http/exceptions.rb +5 -5
  91. data/lib/ronin/network/http/exceptions/unknown_request.rb +5 -5
  92. data/lib/ronin/network/http/http.rb +65 -70
  93. data/lib/ronin/network/http/proxy.rb +5 -5
  94. data/lib/ronin/network/imap.rb +16 -15
  95. data/lib/ronin/network/mixins.rb +6 -5
  96. data/lib/ronin/network/mixins/dns.rb +5 -5
  97. data/lib/ronin/network/mixins/esmtp.rb +5 -5
  98. data/lib/ronin/network/mixins/ftp.rb +155 -0
  99. data/lib/ronin/network/mixins/http.rb +58 -587
  100. data/lib/ronin/network/mixins/imap.rb +5 -5
  101. data/lib/ronin/network/mixins/mixin.rb +5 -5
  102. data/lib/ronin/network/mixins/pop3.rb +5 -5
  103. data/lib/ronin/network/mixins/smtp.rb +5 -5
  104. data/lib/ronin/network/mixins/ssl.rb +5 -5
  105. data/lib/ronin/network/mixins/tcp.rb +43 -10
  106. data/lib/ronin/network/mixins/telnet.rb +5 -5
  107. data/lib/ronin/network/mixins/udp.rb +126 -6
  108. data/lib/ronin/network/mixins/unix.rb +279 -0
  109. data/lib/ronin/network/network.rb +5 -5
  110. data/lib/ronin/network/pop3.rb +10 -10
  111. data/lib/ronin/network/proxy.rb +578 -0
  112. data/lib/ronin/network/smtp.rb +5 -5
  113. data/lib/ronin/network/smtp/email.rb +6 -6
  114. data/lib/ronin/network/smtp/smtp.rb +12 -13
  115. data/lib/ronin/network/ssl.rb +16 -17
  116. data/lib/ronin/network/tcp.rb +7 -310
  117. data/lib/ronin/network/tcp/proxy.rb +417 -0
  118. data/lib/ronin/network/tcp/tcp.rb +452 -0
  119. data/lib/ronin/network/telnet.rb +34 -28
  120. data/lib/ronin/network/udp.rb +7 -271
  121. data/lib/ronin/network/udp/proxy.rb +191 -0
  122. data/lib/ronin/network/udp/udp.rb +452 -0
  123. data/lib/ronin/network/unix.rb +286 -0
  124. data/lib/ronin/path.rb +35 -39
  125. data/lib/ronin/spec/ui/output.rb +6 -12
  126. data/lib/ronin/support.rb +6 -5
  127. data/lib/ronin/support/inflector.rb +8 -12
  128. data/lib/ronin/support/support.rb +7 -5
  129. data/lib/ronin/support/version.rb +6 -6
  130. data/lib/ronin/templates.rb +5 -5
  131. data/lib/ronin/templates/erb.rb +5 -5
  132. data/lib/ronin/templates/template.rb +5 -5
  133. data/lib/ronin/ui/output.rb +5 -5
  134. data/lib/ronin/ui/output/helpers.rb +42 -28
  135. data/lib/ronin/ui/output/output.rb +17 -21
  136. data/lib/ronin/ui/output/terminal.rb +5 -5
  137. data/lib/ronin/ui/output/terminal/color.rb +15 -9
  138. data/lib/ronin/ui/output/terminal/raw.rb +5 -5
  139. data/lib/ronin/ui/shell.rb +8 -11
  140. data/lib/ronin/wordlist.rb +110 -30
  141. data/ronin-support.gemspec +39 -109
  142. data/spec/binary/hexdump/helpers/hexdumps.rb +13 -0
  143. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/ascii.bin +0 -0
  144. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/hexdump_decimal_shorts.txt +0 -0
  145. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/hexdump_hex_bytes.txt +0 -0
  146. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/hexdump_hex_shorts.txt +0 -0
  147. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/hexdump_octal_bytes.txt +0 -0
  148. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/hexdump_octal_shorts.txt +0 -0
  149. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/hexdump_repeated.txt +0 -0
  150. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_decimal_bytes.txt +0 -0
  151. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_decimal_ints.txt +0 -0
  152. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_decimal_quads.txt +0 -0
  153. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_decimal_shorts.txt +0 -0
  154. data/spec/binary/hexdump/helpers/hexdumps/od_doubles.txt +17 -0
  155. data/spec/binary/hexdump/helpers/hexdumps/od_floats.txt +17 -0
  156. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_hex_bytes.txt +0 -0
  157. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_hex_ints.txt +0 -0
  158. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_hex_quads.txt +0 -0
  159. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_hex_shorts.txt +0 -0
  160. data/spec/binary/hexdump/helpers/hexdumps/od_named_chars.txt +17 -0
  161. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_octal_bytes.txt +0 -0
  162. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_octal_ints.txt +0 -0
  163. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_octal_quads.txt +0 -0
  164. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_octal_shorts.txt +0 -0
  165. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/od_repeated.txt +0 -0
  166. data/spec/{formatting/binary → binary/hexdump}/helpers/hexdumps/repeated.bin +0 -0
  167. data/spec/binary/hexdump/parser_spec.rb +302 -0
  168. data/spec/binary/struct_spec.rb +496 -0
  169. data/spec/binary/template_spec.rb +416 -0
  170. data/spec/extensions/enumerable_spec.rb +4 -4
  171. data/spec/extensions/file_spec.rb +12 -14
  172. data/spec/extensions/ip_addr_spec.rb +76 -50
  173. data/spec/extensions/kernel_spec.rb +7 -7
  174. data/spec/extensions/regexp_spec.rb +119 -59
  175. data/spec/extensions/resolv_spec.rb +2 -2
  176. data/spec/extensions/string_spec.rb +31 -30
  177. data/spec/formatting/binary/array_spec.rb +26 -0
  178. data/spec/formatting/binary/base64_spec.rb +50 -0
  179. data/spec/formatting/binary/float_spec.rb +36 -0
  180. data/spec/formatting/binary/integer_spec.rb +76 -50
  181. data/spec/formatting/binary/string_spec.rb +91 -198
  182. data/spec/formatting/digest/string_spec.rb +5 -5
  183. data/spec/formatting/html/integer_spec.rb +6 -6
  184. data/spec/formatting/html/string_spec.rb +10 -10
  185. data/spec/formatting/http/integer_spec.rb +3 -3
  186. data/spec/formatting/http/string_spec.rb +5 -5
  187. data/spec/formatting/sql/string_spec.rb +21 -19
  188. data/spec/formatting/text/array_spec.rb +15 -15
  189. data/spec/formatting/text/string_spec.rb +58 -28
  190. data/spec/fuzzing/extensions/string_spec.rb +87 -0
  191. data/spec/fuzzing/fuzzer_spec.rb +109 -0
  192. data/spec/fuzzing/fuzzing_spec.rb +24 -0
  193. data/spec/fuzzing/mutator_spec.rb +112 -0
  194. data/spec/fuzzing/repeater_spec.rb +57 -0
  195. data/spec/fuzzing/template_spec.rb +54 -0
  196. data/spec/mixin_spec.rb +10 -12
  197. data/spec/network/dns_spec.rb +89 -23
  198. data/spec/network/ftp_spec.rb +81 -0
  199. data/spec/network/http/http_spec.rb +237 -144
  200. data/spec/network/http/proxy_spec.rb +37 -37
  201. data/spec/network/network_spec.rb +2 -2
  202. data/spec/network/proxy_spec.rb +121 -0
  203. data/spec/network/shared/unix_server.rb +31 -0
  204. data/spec/network/smtp/email_spec.rb +14 -14
  205. data/spec/network/ssl_spec.rb +53 -3
  206. data/spec/network/tcp/proxy_spec.rb +118 -0
  207. data/spec/network/tcp/tcp_spec.rb +316 -0
  208. data/spec/network/telnet_spec.rb +67 -0
  209. data/spec/network/udp/udp_spec.rb +298 -0
  210. data/spec/network/unix_spec.rb +182 -0
  211. data/spec/path_spec.rb +43 -18
  212. data/spec/spec_helper.rb +2 -3
  213. data/spec/support/inflector_spec.rb +4 -4
  214. data/spec/support_spec.rb +1 -1
  215. data/spec/templates/erb_spec.rb +3 -3
  216. data/spec/templates/template_spec.rb +10 -10
  217. data/spec/ui/shell_spec.rb +15 -15
  218. data/spec/wordlist_spec.rb +80 -19
  219. metadata +176 -121
  220. data/.gemtest +0 -0
  221. data/spec/formatting/binary/helpers/hexdumps.rb +0 -16
  222. data/spec/fuzzing/string_spec.rb +0 -158
  223. data/spec/network/tcp_spec.rb +0 -247
  224. data/spec/network/udp_spec.rb +0 -248
@@ -0,0 +1,118 @@
1
+ require 'spec_helper'
2
+ require 'ronin/network/tcp/proxy'
3
+
4
+ describe Network::TCP::Proxy, :network => true do
5
+ let(:port) { 1337 }
6
+ let(:host) { 'localhost' }
7
+ let(:server) { ['www.example.com', 80] }
8
+
9
+ before(:each) do
10
+ @proxy = described_class.new(
11
+ :port => port,
12
+ :host => host,
13
+ :server => server
14
+ )
15
+ @thread = Thread.new { @proxy.start }
16
+
17
+ sleep 0.1
18
+ end
19
+
20
+ describe "#on_client_connect" do
21
+ let(:injection) { "Client connected\r\n" }
22
+
23
+ before do
24
+ @proxy.on_client_connect do |client|
25
+ client.write(injection)
26
+ end
27
+
28
+ @socket = TCPSocket.new(host,port)
29
+ end
30
+
31
+ it "should trigger when a new client connects" do
32
+ expect(@socket.readline).to eq(injection)
33
+ end
34
+
35
+ after { @socket.close }
36
+ end
37
+
38
+ describe "#on_server_connect" do
39
+ let(:injection) { "Server connected\r\n" }
40
+
41
+ before do
42
+ @proxy.on_server_connect do |client,server|
43
+ client.write(injection)
44
+ end
45
+
46
+ @socket = TCPSocket.new(host,port)
47
+ end
48
+
49
+ it "should trigger after a new client connects" do
50
+ expect(@socket.readline).to eq(injection)
51
+ end
52
+
53
+ after { @socket.close }
54
+ end
55
+
56
+ describe "#on_client_data" do
57
+ before do
58
+ @proxy.on_client_data do |client,server,data|
59
+ if (match = data.match(/HTTP\/(\d\.\d)/))
60
+ client.write("HTTP #{match[1]} detected!\r\n")
61
+ end
62
+ end
63
+
64
+ @socket = TCPSocket.new(host,port)
65
+ end
66
+
67
+ it "should trigger when the client sends data" do
68
+ @socket.write("GET /placeholder HTTP/1.0\r\n\r\n")
69
+
70
+ expect(@socket.readline).to eq("HTTP 1.0 detected!\r\n")
71
+ end
72
+
73
+ after { @socket.close }
74
+ end
75
+
76
+ describe "#on_server_data" do
77
+ before do
78
+ @proxy.on_server_data do |client,server,data|
79
+ data.gsub!(/Connection: \S+/,'Connection: keep-alive')
80
+ end
81
+
82
+ @socket = TCPSocket.new(host,port)
83
+ end
84
+
85
+ it "should trigger when the server sends data" do
86
+ @socket.write("GET / HTTP/1.0\r\n\r\n")
87
+
88
+ expect(@socket.read).to include("Connection: keep-alive\r\n")
89
+ end
90
+
91
+ after { @socket.close }
92
+ end
93
+
94
+ describe "#on_server_disconnect" do
95
+ let(:injection) { "Haha Internet!\r\n" }
96
+
97
+ before do
98
+ @proxy.on_server_disconnect do |client,server|
99
+ client.write(injection)
100
+ end
101
+
102
+ @socket = TCPSocket.new(host,port)
103
+ end
104
+
105
+ it "should trigger when the server closes the connection" do
106
+ @socket.write("GET / HTTP/1.0\r\n\r\n")
107
+
108
+ expect(@socket.read.end_with?(injection)).to be(true)
109
+ end
110
+
111
+ after { @socket.close }
112
+ end
113
+
114
+ after(:each) do
115
+ @thread.kill
116
+ @proxy.close
117
+ end
118
+ end
@@ -0,0 +1,316 @@
1
+ require 'spec_helper'
2
+ require 'ronin/network/tcp'
3
+
4
+ require 'resolv'
5
+
6
+ describe Network::TCP do
7
+ describe "helper methods", :network do
8
+ let(:host) { 'smtp.gmail.com' }
9
+ let(:port) { 587 }
10
+
11
+ subject do
12
+ obj = Object.new
13
+ obj.extend described_class
14
+ obj
15
+ end
16
+
17
+ shared_examples "TCP Server" do
18
+ let(:server_host) { 'localhost' }
19
+ let(:server_port) { 1024 + rand(65535 - 1024) }
20
+ let(:server_ip) { Resolv.getaddress(server_host) }
21
+ let(:server) { TCPServer.new(server_host,server_port) }
22
+ let(:server_bind_ip) { server.addr[3] }
23
+ let(:server_bind_port) { server.addr[1] }
24
+
25
+ before(:each) { server.listen(1) }
26
+ after(:each) { server.close }
27
+ end
28
+
29
+ describe "#tcp_open?" do
30
+ include_examples "TCP Server"
31
+
32
+ let(:host) { server_bind_ip }
33
+ let(:port) { server_bind_port }
34
+
35
+ it "should return true for open ports" do
36
+ expect(subject.tcp_open?(host,port)).to be(true)
37
+ end
38
+
39
+ let(:closed_port) { port + 1 }
40
+
41
+ it "should return false for closed ports" do
42
+ expect(subject.tcp_open?(host,closed_port)).to be(false)
43
+ end
44
+
45
+ context "when given a timeout" do
46
+ it "should have a timeout for firewalled ports" do
47
+ timeout = 2
48
+
49
+ t1 = Time.now
50
+ subject.tcp_open?(host,port+1,nil,nil,timeout)
51
+ t2 = Time.now
52
+
53
+ expect((t2 - t1).to_i).to be <= timeout
54
+ end
55
+ end
56
+ end
57
+
58
+ describe "#tcp_connect" do
59
+ let(:host) { 'example.com' }
60
+ let(:port) { 80 }
61
+
62
+ it "should open a TCPSocket" do
63
+ socket = subject.tcp_connect(host,port)
64
+
65
+ expect(socket).to be_kind_of(TCPSocket)
66
+ expect(socket).not_to be_closed
67
+
68
+ socket.close
69
+ end
70
+
71
+ context "when given a local host and port" do
72
+ let(:local_port) { 1024 + rand(65535 - 1024) }
73
+
74
+ it "should bind to a local host and port" do
75
+ socket = subject.tcp_connect(host,port,nil,local_port)
76
+ bound_port = socket.addr[1]
77
+
78
+ expect(bound_port).to eq(local_port)
79
+
80
+ socket.close
81
+ end
82
+ end
83
+
84
+ context "when given a block" do
85
+ it "should yield the new TCPSocket" do
86
+ socket = nil
87
+
88
+ subject.tcp_connect(host,port) do |yielded_socket|
89
+ socket = yielded_socket
90
+ end
91
+
92
+ expect(socket).not_to be_closed
93
+ socket.close
94
+ end
95
+ end
96
+ end
97
+
98
+ describe "#tcp_connect_and_send" do
99
+ let(:data) { "HELO ronin-support\n" }
100
+
101
+ let(:expected_response) { "250 #{host} at your service\r\n" }
102
+
103
+ it "should connect and then send data" do
104
+ socket = subject.tcp_connect_and_send(data,host,port)
105
+ banner = socket.readline
106
+ response = socket.readline
107
+
108
+ expect(response).to eq(expected_response)
109
+
110
+ socket.close
111
+ end
112
+
113
+ context "when given a local host and port" do
114
+ let(:local_port) { 1024 + rand(65535 - 1024) }
115
+
116
+ it "should bind to a local host and port" do
117
+ socket = subject.tcp_connect_and_send(data,host,port,nil,local_port)
118
+ bound_port = socket.addr[1]
119
+
120
+ expect(bound_port).to eq(local_port)
121
+
122
+ socket.close
123
+ end
124
+ end
125
+
126
+ context "when given a block" do
127
+ it "should yield the TCPSocket" do
128
+ response = nil
129
+
130
+ socket = subject.tcp_connect_and_send(data,host,port) do |socket|
131
+ banner = socket.readline
132
+ response = socket.readline
133
+ end
134
+
135
+ expect(response).to eq(expected_response)
136
+
137
+ socket.close
138
+ end
139
+ end
140
+ end
141
+
142
+ describe "#tcp_session" do
143
+ it "should open then close a TCPSocket" do
144
+ socket = nil
145
+
146
+ subject.tcp_session(host,port) do |yielded_socket|
147
+ socket = yielded_socket
148
+ end
149
+
150
+ expect(socket).to be_kind_of(TCPSocket)
151
+ expect(socket).to be_closed
152
+ end
153
+
154
+ context "when given a local host and port" do
155
+ let(:local_port) { 1024 + rand(65535 - 1024) }
156
+
157
+ it "should bind to a local host and port" do
158
+ bound_port = nil
159
+
160
+ subject.tcp_session(host,port,nil,local_port) do |socket|
161
+ bound_port = socket.addr[1]
162
+ end
163
+
164
+ expect(bound_port).to eq(local_port)
165
+ end
166
+ end
167
+ end
168
+
169
+ describe "#tcp_banner" do
170
+ let(:host) { 'smtp.gmail.com' }
171
+ let(:port) { 587 }
172
+
173
+ let(:expected_banner) { /^220 #{Regexp.escape(host)} ESMTP [^\s]+ - gsmtp$/ }
174
+
175
+ it "should return the read service banner" do
176
+ banner = subject.tcp_banner(host,port)
177
+
178
+ expect(banner).to match(expected_banner)
179
+ end
180
+
181
+ context "when given a local host and port" do
182
+ let(:local_port) { 1024 + rand(65535 - 1024) }
183
+
184
+ it "should bind to a local host and port" do
185
+ banner = subject.tcp_banner(host,port,nil,local_port)
186
+
187
+ expect(banner).to match(expected_banner)
188
+ end
189
+ end
190
+
191
+ context "when given a block" do
192
+ it "should yield the banner" do
193
+ banner = nil
194
+
195
+ subject.tcp_banner(host,port) do |yielded_banner|
196
+ banner = yielded_banner
197
+ end
198
+
199
+ expect(banner).to match(expected_banner)
200
+ end
201
+ end
202
+ end
203
+
204
+ let(:local_host) { 'localhost' }
205
+ let(:local_ip) { Resolv.getaddress(local_host) }
206
+
207
+ describe "#tcp_send" do
208
+ include_context "TCP Server"
209
+
210
+ let(:data) { "hello\n" }
211
+
212
+ it "should send data to a service" do
213
+ subject.tcp_send(data,server_bind_ip,server_bind_port)
214
+
215
+ client = server.accept
216
+ sent = client.readline
217
+
218
+ client.close
219
+
220
+ expect(sent).to eq(data)
221
+ end
222
+
223
+ context "when given a local host and port" do
224
+ let(:local_port) { 1024 + rand(65535 - 1024) }
225
+
226
+ it "should bind to a local host and port" do
227
+ subject.tcp_send(data,server_bind_ip,server_bind_port,server_bind_ip,local_port)
228
+
229
+ client = server.accept
230
+ client_port = client.peeraddr[1]
231
+
232
+ expect(client_port).to eq(local_port)
233
+
234
+ client.close
235
+ end
236
+ end
237
+ end
238
+
239
+ describe "#tcp_server" do
240
+ it "should create a new TCPServer" do
241
+ server = subject.tcp_server
242
+
243
+ expect(server).to be_kind_of(TCPServer)
244
+ expect(server).not_to be_closed
245
+
246
+ server.close
247
+ end
248
+
249
+ context "when given a local host and port" do
250
+ let(:local_port) { 1024 + rand(65535 - 1024) }
251
+
252
+ it "should bind to a specific port and host" do
253
+ server = subject.tcp_server(local_port,local_host)
254
+ bound_host = server.addr[3]
255
+ bound_port = server.addr[1]
256
+
257
+ expect(bound_host).to eq(local_ip)
258
+ expect(bound_port).to eq(local_port)
259
+
260
+ server.close
261
+ end
262
+ end
263
+
264
+ context "when given a block" do
265
+ it "should yield the new TCPServer" do
266
+ server = nil
267
+
268
+ subject.tcp_server do |yielded_server|
269
+ server = yielded_server
270
+ end
271
+
272
+ expect(server).to be_kind_of(TCPServer)
273
+ expect(server).not_to be_closed
274
+
275
+ server.close
276
+ end
277
+ end
278
+ end
279
+
280
+ describe "#tcp_server_session" do
281
+ it "should create a temporary TCPServer" do
282
+ server = nil
283
+
284
+ subject.tcp_server_session do |yielded_server|
285
+ server = yielded_server
286
+ end
287
+
288
+ expect(server).to be_kind_of(TCPServer)
289
+ expect(server).to be_closed
290
+ end
291
+
292
+ context "when given a block" do
293
+ let(:local_port) { 1024 + rand(65535 - 1024) }
294
+
295
+ it "should bind to a specific port and host" do
296
+ bound_host = nil
297
+ bound_port = nil
298
+
299
+ subject.tcp_server_session(local_port,local_host) do |server|
300
+ bound_host = server.addr[3]
301
+ bound_port = server.addr[1]
302
+ end
303
+
304
+ expect(bound_host).to eq(local_ip)
305
+ expect(bound_port).to eq(local_port)
306
+ end
307
+ end
308
+ end
309
+
310
+ describe "#tcp_accept" do
311
+ let(:server_port) { 1024 + rand(65535 - 1024) }
312
+
313
+ pending "need to automate connecting to the TCPServer"
314
+ end
315
+ end
316
+ end
@@ -0,0 +1,67 @@
1
+ require 'spec_helper'
2
+ require 'ronin/network/telnet'
3
+
4
+ describe Network::Telnet do
5
+ describe "helpers", :network do
6
+ let(:host) { 'towel.blinkenlights.nl' }
7
+
8
+ subject do
9
+ obj = Object.new
10
+ obj.extend described_class
11
+ obj
12
+ end
13
+
14
+ describe "#telnet_connect" do
15
+ it "should return a Net::Telnet object" do
16
+ telnet = subject.telnet_connect(host)
17
+
18
+ expect(telnet).to be_kind_of(Net::Telnet)
19
+ telnet.close
20
+ end
21
+
22
+ it "should connect to a telnet service on port 23" do
23
+ telnet = subject.telnet_connect(host)
24
+
25
+ telnet.close
26
+ end
27
+
28
+ context "when given a block" do
29
+ it "should yield the new Net::Telnet object" do
30
+ telnet = nil
31
+
32
+ subject.telnet_connect(host) do |telnet_object|
33
+ telnet = telnet_object
34
+ end
35
+
36
+ expect(telnet).to be_kind_of(Net::Telnet)
37
+ telnet.close
38
+ end
39
+ end
40
+ end
41
+
42
+ describe "#telnet_session" do
43
+ it "should yield a new Net::Telnet object" do
44
+ yielded_telnet = nil
45
+
46
+ subject.telnet_session(host) do |telnet|
47
+ yielded_telnet = telnet
48
+ end
49
+
50
+ expect(yielded_telnet).to be_kind_of(Net::Telnet)
51
+ end
52
+
53
+ it "should close the Telnet session after yielding it" do
54
+ session = nil
55
+ was_open = nil
56
+
57
+ subject.telnet_session(host) do |telnet|
58
+ session = telnet
59
+ was_open = !telnet.sock.closed?
60
+ end
61
+
62
+ expect(was_open).to be(true)
63
+ expect(session.sock).to be_closed
64
+ end
65
+ end
66
+ end
67
+ end