Line data Source code
1 : #include "includes.h"
2 : #include "version.h"
3 : #include "libcli/libcli.h"
4 : #include "lib/events/events.h"
5 : #include "libcli/resolve/resolve.h"
6 : #include "param/param.h"
7 : #include "libcli/raw/raw_proto.h"
8 : #include "libcli/http/http.h"
9 : #include "credentials.h"
10 : #include "util/tevent_ntstatus.h"
11 : #include "lib/tls/tls.h"
12 : #include "lib/cmdline/cmdline.h"
13 :
14 :
15 : struct http_client_info {
16 : struct http_conn *http_conn;
17 : uint16_t server_port;
18 : const char *server_addr;
19 : struct tstream_tls_params *tls_params;
20 : struct cli_credentials *creds;
21 : struct loadparm_context *lp_ctx;
22 : const char *uri;
23 : };
24 :
25 14 : static bool send_http_request(TALLOC_CTX *mem_ctx,
26 : struct tevent_context *ev_ctx,
27 : struct http_client_info* es,
28 : size_t response_size,
29 : NTSTATUS *pstatus)
30 : {
31 14 : struct http_request *http_req = NULL;
32 14 : struct tevent_req *req = NULL;
33 14 : char *uri = NULL;
34 14 : struct http_request *http_response = NULL;
35 : NTSTATUS status;
36 :
37 14 : http_req = talloc_zero(mem_ctx, struct http_request);
38 14 : if (!http_req) {
39 0 : DBG_ERR("no memory\n");
40 0 : return false;
41 : }
42 :
43 14 : uri = talloc_strdup(mem_ctx, es->uri);
44 :
45 14 : http_req->type = HTTP_REQ_POST;
46 14 : http_req->uri = uri;
47 14 : http_req->body = data_blob_null;
48 14 : http_req->major = '1';
49 14 : http_req->minor = '1';
50 :
51 14 : http_add_header(mem_ctx, &http_req->headers,
52 : "User-Agent", "Samba/http_test");
53 14 : http_add_header(mem_ctx, &http_req->headers,
54 : "Accept", "*/*");
55 :
56 14 : req = http_send_auth_request_send(mem_ctx,
57 : ev_ctx,
58 : es->http_conn,
59 : http_req,
60 : es->creds,
61 : es->lp_ctx,
62 : HTTP_AUTH_BASIC);
63 14 : if (!tevent_req_set_endtime(req, ev_ctx, timeval_current_ofs(10, 0))) {
64 0 : DBG_ERR("Failed to set timeout\n");
65 0 : return false;
66 : }
67 :
68 14 : if (!tevent_req_poll_ntstatus(req, ev_ctx, pstatus)) {
69 0 : DBG_ERR("Failed to connect: %s\n", nt_errstr(*pstatus));
70 0 : return false;
71 : }
72 :
73 14 : status = http_send_auth_request_recv(req);
74 14 : if (!NT_STATUS_IS_OK(status)) {
75 0 : DBG_ERR("Auth request failed: %s\n", nt_errstr(status));
76 0 : return false;
77 : }
78 :
79 14 : req = http_read_response_send(mem_ctx,
80 : ev_ctx,
81 : es->http_conn,
82 : response_size);
83 14 : if (!req) {
84 0 : DBG_ERR("no memory\n");
85 0 : return -1;
86 : }
87 :
88 14 : if (!tevent_req_set_endtime(req, ev_ctx, timeval_current_ofs(10, 0))) {
89 0 : DBG_ERR("Failed to set timeout\n");
90 0 : return false;
91 : }
92 :
93 14 : if (!tevent_req_poll_ntstatus(req, ev_ctx, pstatus)) {
94 0 : DBG_ERR("Failed to read_resonse: %s\n", nt_errstr(*pstatus));
95 0 : return false;
96 : }
97 :
98 14 : *pstatus = http_read_response_recv(req, mem_ctx, &http_response);
99 :
100 14 : if (!NT_STATUS_IS_OK(*pstatus)) {
101 2 : DBG_ERR("Failed to receive response: %s\n", nt_errstr(*pstatus));
102 2 : return false;
103 : }
104 : /* following are not 'hard' errors */
105 12 : if (http_response->response_code != 200) {
106 0 : fprintf(stdout, "HTTP server response: %u\n",
107 0 : http_response->response_code);
108 0 : fflush(stdout);
109 0 : return false;
110 :
111 : }
112 12 : if (http_response->body.length == 0) {
113 4 : fprintf(stdout, "unexpected 0 len response\n");
114 4 : fflush(stdout);
115 4 : return false;
116 : }
117 8 : DBG_ERR("response: len (%d)\n%s\n",
118 : (int)http_response->body.length,
119 : talloc_strndup(mem_ctx,
120 : (char *)http_response->body.data,
121 : http_response->body.length));
122 8 : fprintf(stdout,"%s", talloc_strndup(mem_ctx,
123 8 : (char *)http_response->body.data,
124 8 : http_response->body.length));
125 8 : fflush(stdout);
126 8 : return true;
127 : }
128 :
129 14 : int main(int argc, const char *argv[])
130 :
131 : {
132 : TALLOC_CTX *mem_ctx;
133 : struct tevent_context *ev_ctx;
134 14 : int retries = 4;
135 14 : int count = 0;
136 14 : struct http_client_info *http_info = NULL;
137 14 : bool use_tls = false;
138 : int res;
139 : NTSTATUS status;
140 14 : struct tevent_req *req = NULL;
141 14 : bool connected = false;
142 : poptContext pc;
143 14 : const char **const_argv = discard_const_p(const char *, argv);
144 : int opt;
145 : bool ok;
146 14 : const char *ca_file = NULL;
147 14 : int port = 0;
148 14 : size_t response_size = 8192000;
149 : struct cli_credentials *cli_creds;
150 :
151 42 : struct poptOption long_options[] = {
152 : POPT_AUTOHELP
153 :
154 : {
155 : .longName = "usetls",
156 : .shortName = 't',
157 : .argInfo = POPT_ARG_NONE,
158 : .arg = NULL,
159 : .val = 't',
160 : .descrip = "Use tls",
161 : .argDescrip = "enable tls",
162 : },
163 : {
164 : .longName = "ip-address",
165 : .shortName = 'I',
166 : .argInfo = POPT_ARG_STRING,
167 : .arg = NULL,
168 : .val = 'I',
169 : .descrip = "Use this IP to connect to",
170 : .argDescrip = "IP",
171 : },
172 : {
173 : .longName = "port",
174 : .shortName = 'p',
175 : .argInfo = POPT_ARG_INT,
176 : .arg = &port,
177 : .val = 'p',
178 : .descrip = "port to connect to",
179 : .argDescrip = "port",
180 : },
181 : {
182 : .longName = "cacart",
183 : .shortName = 'c',
184 : .argInfo = POPT_ARG_STRING,
185 : .arg = NULL,
186 : .val = 'c',
187 : .descrip = "CA certificate to verify peer against",
188 : .argDescrip = "ca cert",
189 : },
190 : {
191 : .longName = "uri",
192 : .shortName = 'u',
193 : .argInfo = POPT_ARG_STRING,
194 : .arg = NULL,
195 : .val = 'u',
196 : .descrip = "uri to send as part of http request",
197 : .argDescrip = "uri",
198 : },
199 : {
200 : .longName = "rsize",
201 : .argInfo = POPT_ARG_LONG,
202 : .arg = &response_size,
203 : .descrip = "response size",
204 : },
205 14 : POPT_COMMON_SAMBA
206 14 : POPT_COMMON_CREDENTIALS
207 : POPT_TABLEEND
208 : };
209 :
210 14 : mem_ctx = talloc_init("http_test");
211 :
212 14 : if (!mem_ctx) {
213 0 : DBG_ERR("Not enough memory\n");
214 0 : res = -1;
215 0 : goto done;
216 : }
217 :
218 14 : http_info = talloc_zero(mem_ctx, struct http_client_info);
219 :
220 14 : if (http_info == NULL) {
221 0 : DBG_ERR("Not enough memory\n");
222 0 : res = -1;
223 0 : goto done;
224 : }
225 :
226 14 : ok = samba_cmdline_init(mem_ctx,
227 : SAMBA_CMDLINE_CONFIG_CLIENT,
228 : false /* require_smbconf */);
229 14 : if (!ok) {
230 0 : DBG_ERR("Failed to init cmdline parser!\n");
231 0 : res = -1;
232 0 : goto done;
233 : }
234 :
235 14 : pc = samba_popt_get_context(getprogname(),
236 : argc,
237 : const_argv,
238 : long_options,
239 : 0);
240 14 : if (pc == NULL) {
241 0 : DBG_ERR("Failed to setup popt context!\n");
242 0 : res = -1;
243 0 : goto done;
244 : }
245 :
246 : /* some defaults */
247 :
248 14 : http_info->server_addr = "localhost";
249 14 : http_info->uri = "/_search?pretty";
250 :
251 42 : while ((opt = poptGetNextOpt(pc)) != -1) {
252 28 : switch (opt) {
253 0 : case 't':
254 0 : use_tls = true;
255 0 : break;
256 0 : case 'c': {
257 0 : ca_file = talloc_strdup(mem_ctx,
258 0 : poptGetOptArg(pc));
259 0 : if (ca_file == NULL) {
260 0 : DBG_ERR("Not enough memory\n");
261 0 : res = -1;
262 0 : goto done;
263 : }
264 0 : break;
265 : }
266 14 : case 'I': {
267 14 : http_info->server_addr = talloc_strdup(mem_ctx,
268 14 : poptGetOptArg(pc));
269 14 : if (http_info->server_addr == NULL) {
270 0 : DBG_ERR("Not enough memory\n");
271 0 : res = -1;
272 0 : goto done;
273 : }
274 14 : break;
275 : }
276 14 : case 'u': {
277 14 : http_info->uri = talloc_strdup(mem_ctx,
278 14 : poptGetOptArg(pc));
279 14 : if (http_info->uri == NULL) {
280 0 : DBG_ERR("Not enough memory\n");
281 0 : res = -1;
282 0 : goto done;
283 : }
284 14 : break;
285 : }
286 : }
287 : }
288 :
289 14 : if (use_tls && ca_file == NULL) {
290 0 : DBG_ERR("No cacert\n");
291 0 : res = -1;
292 0 : poptPrintUsage(pc, stderr, 0);
293 0 : goto done;
294 : }
295 :
296 14 : if (!port) {
297 14 : port = 8080;
298 : }
299 14 : http_info->server_port = port;
300 :
301 14 : ev_ctx = s4_event_context_init(mem_ctx);
302 14 : if (!ev_ctx) {
303 0 : DBG_ERR("Not enough memory\n");
304 0 : res = -1;
305 0 : goto done;
306 : }
307 :
308 :
309 14 : cli_creds = samba_cmdline_get_creds();
310 14 : if (!cli_credentials_is_anonymous(cli_creds)) {
311 0 : http_info->creds = cli_credentials_init(mem_ctx);
312 0 : cli_credentials_set_username(
313 : http_info->creds,
314 : cli_credentials_get_username(cli_creds),
315 : CRED_SPECIFIED);
316 0 : cli_credentials_set_password(http_info->creds,
317 : cli_credentials_get_password(cli_creds),
318 : CRED_SPECIFIED);
319 : } else {
320 14 : DBG_DEBUG("Anonymous creds!!!\n");
321 14 : http_info->creds = cli_creds;
322 : }
323 14 : if (http_info->creds == NULL) {
324 0 : DBG_ERR("Failed to create creds\n");
325 0 : res = -1;
326 0 : goto done;
327 : }
328 14 : http_info->lp_ctx = samba_cmdline_get_lp_ctx();
329 :
330 14 : DBG_ERR("retries = %d/%d, Using server %s, port %d, using tls %s\n",
331 : count, retries,
332 : http_info->server_addr,
333 : http_info->server_port,
334 : use_tls ? "true" : "false");
335 :
336 14 : while (count < retries) {
337 : int error;
338 14 : DBG_ERR("Connecting to HTTP [%s] port [%"PRIu16"]%s\n",
339 : http_info->server_addr, http_info->server_port,
340 : use_tls ? " with tls" : " without tls");
341 14 : if (use_tls) {
342 0 : bool system_cas = false;
343 0 : const char * const *ca_dirs = NULL;
344 0 : const char *crl_file = NULL;
345 0 : const char *tls_priority = "NORMAL:-VERS-SSL3.0";
346 0 : enum tls_verify_peer_state verify_peer =
347 : TLS_VERIFY_PEER_CA_ONLY;
348 :
349 0 : status = tstream_tls_params_client(mem_ctx,
350 : system_cas,
351 : ca_dirs,
352 : ca_file,
353 : crl_file,
354 : tls_priority,
355 : verify_peer,
356 : http_info->server_addr,
357 : &http_info->tls_params);
358 0 : if (!NT_STATUS_IS_OK(status)) {
359 0 : DBG_ERR("Failed tstream_tls_params_client - %s\n",
360 : nt_errstr(status));
361 0 : res = -1;
362 0 : goto done;
363 : }
364 : }
365 :
366 14 : req = http_connect_send(mem_ctx,
367 : ev_ctx,
368 : http_info->server_addr,
369 14 : http_info->server_port,
370 : http_info->creds,
371 : http_info->tls_params);
372 14 : if (!tevent_req_poll_ntstatus(req, ev_ctx, &status)) {
373 0 : res = -1;
374 0 : goto done;
375 : }
376 :
377 14 : error = http_connect_recv(req,
378 : mem_ctx,
379 : &http_info->http_conn);
380 14 : if (error != 0) {
381 0 : count++;
382 0 : DBG_ERR("HTTP connection failed retry %d/%d: %s\n", count, retries, strerror(error));
383 : } else {
384 14 : DBG_ERR("HTTP connection succeeded\n");
385 14 : connected = true;
386 14 : break;
387 : }
388 : }
389 :
390 14 : if (!connected) {
391 0 : DBG_ERR("Leaving early\n");
392 0 : res = -1;
393 0 : goto done;
394 : }
395 :
396 14 : if (!send_http_request(mem_ctx, ev_ctx, http_info, response_size, &status)) {
397 6 : DBG_ERR("Failure\n");
398 6 : res = -1;
399 6 : goto done;
400 : }
401 8 : res = 0;
402 14 : done:
403 14 : TALLOC_FREE(mem_ctx);
404 14 : return res;
405 : }
|