You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

191 lines
4.8 KiB

/*****************************************************************************
* tunnel_test.c: HTTP CONNECT
*****************************************************************************
* Copyright (C) 2015 Rémi Denis-Courmont
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation; either version 2.1 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*****************************************************************************/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#undef NDEBUG
#include <assert.h>
#include <stdbool.h>
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
#ifdef HAVE_SYS_SOCKET_H
#include <sys/socket.h>
#endif
#ifndef SOCK_CLOEXEC
# define SOCK_CLOEXEC 0
# define accept4(a,b,c,d) accept(a,b,c)
#endif
#ifdef _WIN32
# include <winsock2.h>
#else
# include <netinet/in.h>
#endif
#ifdef HAVE_ARPA_INET_H
#include <arpa/inet.h>
#endif
#include <vlc_common.h>
#include <vlc_threads.h>
#include <vlc_tls.h>
#include "transport.h"
const char vlc_module_name[] = "test_http_tunnel";
static void proxy_client_process(int fd)
{
char buf[1024];
size_t buflen = 0;
ssize_t val;
/* Read request (and possibly more) */
while (strnstr(buf, "\r\n\r\n", buflen) == NULL)
{
val = recv(fd, buf + buflen, sizeof (buf) - buflen - 1, 0);
if (val <= 0)
assert(!"Incomplete request");
buflen += val;
}
buf[buflen] = '\0';
char host[64];
unsigned port, ver;
int offset;
assert(sscanf(buf, "CONNECT %63[^:]:%u HTTP/1.%1u%n", host, &port, &ver,
&offset) == 3);
assert(!strcmp(host, "www.example.com"));
assert(port == 443);
assert(ver == 1);
assert(sscanf(buf + offset + 2, "Host: %63[^:]:%u", host, &port) == 2);
assert(!strcmp(host, "www.example.com"));
assert(port == 443);
assert(strstr(buf, "\r\nProxy-Authorization: Basic "
"QWxhZGRpbjpvcGVuIHNlc2FtZQ==\r\n") != NULL);
const char resp[] = "HTTP/1.1 500 Failure\r\n\r\n";
val = write(fd, resp, strlen(resp));
assert((size_t)val == strlen(resp));
shutdown(fd, SHUT_WR);
}
static unsigned connection_count = 0;
static void *proxy_thread(void *data)
{
int *lfd = data;
for (;;)
{
int cfd = accept4(*lfd, NULL, NULL, SOCK_CLOEXEC);
if (cfd == -1)
continue;
int canc = vlc_savecancel();
proxy_client_process(cfd);
vlc_close(cfd);
connection_count++;
vlc_restorecancel(canc);
}
vlc_assert_unreachable();
}
static int server_socket(unsigned *port)
{
int fd = socket(PF_INET6, SOCK_STREAM|SOCK_CLOEXEC, IPPROTO_TCP);
if (fd == -1)
return -1;
vlc_sockaddr addr = {
.sin6.sin6_family = AF_INET6,
#ifdef HAVE_SA_LEN
.sin6.sin6_len = sizeof (addr),
#endif
.sin6.sin6_addr = in6addr_loopback,
};
socklen_t addrlen = sizeof (addr.sin6);
if (bind(fd, &addr.sa, addrlen)
|| getsockname(fd, &addr.sa, &addrlen))
{
vlc_close(fd);
return -1;
}
*port = ntohs(addr.sin6.sin6_port);
return fd;
}
int main(void)
{
char *url;
unsigned port;
bool two = false;
/* Test bad URLs */
vlc_https_connect_proxy(NULL, NULL, "www.example.com", 0, &two,
"/test");
vlc_https_connect_proxy(NULL, NULL, "www.example.com", 0, &two,
"ftp://proxy.example.com/");
int *lfd = malloc(sizeof (int));
assert(lfd != NULL);
*lfd = server_socket(&port);
if (*lfd == -1)
return 77;
if (asprintf(&url, "http://Aladdin:open%%20sesame@[::1]:%u", port) < 0)
url = NULL;
assert(url != NULL);
/* Test connection failure */
vlc_https_connect_proxy(NULL, NULL, "www.example.com", 0, &two, url);
if (listen(*lfd, 255))
{
vlc_close(*lfd);
return 77;
}
vlc_thread_t th;
if (vlc_clone(&th, proxy_thread, lfd))
assert(!"Thread error");
/* Test proxy error */
vlc_https_connect_proxy(NULL, NULL, "www.example.com", 0, &two, url);
vlc_cancel(th);
vlc_join(th, NULL);
assert(connection_count > 0);
free(url);
vlc_close(*lfd);
free(lfd);
return 0;
}