summaryrefslogtreecommitdiff
path: root/tools
diff options
context:
space:
mode:
authorMarcel Holtmann <marcel@holtmann.org>2010-01-06 01:35:48 -0800
committerMarcel Holtmann <marcel@holtmann.org>2010-01-06 01:35:48 -0800
commit1e2882602a72f41a19086a7b0f9d2116200900d7 (patch)
tree67f507de399c2a2f94583e48d8b11cd99fbbb949 /tools
parent50218f3a438e75f7cafa4e7dbaf37b1d1387f47e (diff)
downloadconnman-1e2882602a72f41a19086a7b0f9d2116200900d7.tar.gz
connman-1e2882602a72f41a19086a7b0f9d2116200900d7.tar.bz2
connman-1e2882602a72f41a19086a7b0f9d2116200900d7.zip
Add first draft of a portal test application
Diffstat (limited to 'tools')
-rw-r--r--tools/portal-test.c315
1 files changed, 315 insertions, 0 deletions
diff --git a/tools/portal-test.c b/tools/portal-test.c
new file mode 100644
index 00000000..6aac971e
--- /dev/null
+++ b/tools/portal-test.c
@@ -0,0 +1,315 @@
+/*
+ *
+ * Connection Manager
+ *
+ * Copyright (C) 2007-2010 Intel Corporation. All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ *
+ * 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 General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <errno.h>
+#include <stdio.h>
+#include <unistd.h>
+#include <stdlib.h>
+#include <string.h>
+#include <netdb.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
+
+#include <glib.h>
+
+#define PORT 80
+#define PAGE "/"
+#define USER_APP "connman"
+
+#define CONNECT_TIMEOUT 120
+
+enum get_page_status {
+ GET_PAGE_SUCCESS = 0,
+ GET_PAGE_TIMEOUT = 1,
+ GET_PAGE_FAILED = 2,
+};
+
+struct server_data {
+ char host[80];
+ char page[80];
+ GIOChannel *channel;
+ guint watch;
+ guint timeout;
+ int connection_ready;
+ int sock;
+ int (*get_page) (struct server_data *data, char *page, int len,
+ enum get_page_status status);
+};
+
+static GMainLoop *main_loop = NULL;
+
+static int create_socket()
+{
+ int sk;
+
+ sk = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
+ if (sk < 0) {
+ perror("Error: Can not create TCP socket");
+ exit(1);
+ }
+
+ return sk;
+}
+
+static char *get_ip_from_host(char *host)
+{
+ int ip_len = 15;
+ char *ip;
+ struct hostent *host_ent;
+
+ ip = malloc(ip_len + 1);
+ memset(ip, 0, ip_len + 1);
+ if ((host_ent = gethostbyname(host)) == NULL) {
+ perror("Error: Can not get IP");
+ exit(1);
+ }
+
+ fprintf(stderr, "inside get_ip_from_host %d\n", 2);
+ if (inet_ntop(AF_INET, (void *) host_ent->h_addr_list[0],
+ ip, ip_len) == NULL) {
+ perror("Error: Can not resolve host");
+ exit(1);
+ }
+
+ return ip;
+}
+
+static char *build_get_query(char *host, char *page)
+{
+ char *query;
+ char *host_page = page;
+ char *tpl = "GET /%s HTTP/1.0\r\nHost: %s\r\nUser-Agent: %s\r\n\r\n";
+
+ if(host_page[0] == '/')
+ host_page = host_page + 1;
+
+ query = malloc(strlen(host) + strlen(host_page) +
+ strlen(USER_APP) + strlen(tpl) - 5);
+ sprintf(query, tpl, host_page, host, USER_APP);
+
+ return query;
+}
+
+static gboolean connect_timeout(gpointer user_data)
+{
+ struct server_data *data = user_data;
+
+ data->timeout = 0;
+
+ if (data->get_page)
+ data->get_page(data, NULL, 0, GET_PAGE_TIMEOUT);
+
+ return FALSE;
+}
+
+static void remove_timeout(struct server_data *data)
+{
+ if (data->timeout > 0) {
+ g_source_remove(data->timeout);
+ data->timeout = 0;
+ }
+}
+
+static gboolean tcp_event(GIOChannel *channel, GIOCondition condition,
+ gpointer user_data)
+{
+ char buf[BUFSIZ+1];
+ int len;
+ int sk;
+ struct server_data *data = user_data;
+
+ remove_timeout(data);
+ if (condition & (G_IO_NVAL | G_IO_ERR | G_IO_HUP)) {
+ data->watch = 0;
+ if (data->get_page)
+ data->get_page(data, NULL, 0, GET_PAGE_FAILED);
+
+ return FALSE;
+ }
+
+ sk = g_io_channel_unix_get_fd(channel);
+ len = recv(sk, buf, BUFSIZ, 0);
+
+ if (len > 0) {
+ if (data->get_page)
+ data->get_page(data, buf, len, GET_PAGE_SUCCESS);
+ }
+
+ return TRUE;
+}
+
+static gboolean socket_event(GIOChannel *channel, GIOCondition condition,
+ gpointer user_data)
+{
+ struct server_data *data = user_data;
+ char *query;
+ int sk;
+ unsigned int send_counter = 0;
+ int ret;
+
+ if (condition & G_IO_OUT && data->connection_ready == 0) {
+ data->connection_ready = 1;
+ sk = g_io_channel_unix_get_fd(channel);
+
+ query = build_get_query(data->host, data->page);
+ fprintf(stderr, "qury is:\n%s\n", query);
+
+ while (send_counter < strlen(query)) {
+ ret = send(sk, query+send_counter,
+ strlen(query) - send_counter, 0);
+ if(ret == -1) {
+ perror("Error sending query");
+ remove_timeout(data);
+ if (data->get_page)
+ data->get_page(data, NULL, 0,
+ GET_PAGE_FAILED);
+ free(query);
+ return FALSE;
+ }
+ send_counter += ret;
+ }
+ free(query);
+ fprintf(stdout, "%s \n", "Connection ready");
+ } else if (condition & G_IO_IN)
+ tcp_event(channel, condition, user_data);
+
+ return TRUE;
+}
+
+static void remove_connection(struct server_data *data)
+{
+ remove_timeout(data);
+ g_source_remove(data->watch);
+ g_io_channel_shutdown(data->channel, TRUE, NULL);
+
+ if (data->sock >= 0)
+ close(data->sock);
+
+ g_free(data);
+}
+
+static int get_html(struct server_data *data, int ms_time)
+{
+ struct sockaddr_in *remote_host;
+ int ret;
+ char *ip;
+
+ data->connection_ready = 0;
+ fprintf(stderr, "create socket %d\n", 1);
+ data->sock = create_socket();
+ fprintf(stderr, "call get ip %d\n", 1);
+ ip = get_ip_from_host(data->host);
+ fprintf(stderr, "IP from host %s is %s\n", data->host, ip);
+
+ remote_host = g_try_new0(struct sockaddr_in, 1);
+ remote_host->sin_family = AF_INET;
+ ret = inet_pton(AF_INET, ip, (void *) (&(remote_host->sin_addr.s_addr)));
+ if (ret < 0) {
+ perror("Error Calling inet_pton");
+ goto error;
+ } else if (ret == 0) {
+ fprintf(stderr, "Error: wrong IP address:%s\n", ip);
+ goto error;
+ }
+ remote_host->sin_port = htons(PORT);
+
+ data->channel = g_io_channel_unix_new(data->sock);
+ g_io_channel_set_flags(data->channel, G_IO_FLAG_NONBLOCK, NULL);
+ g_io_channel_set_close_on_unref(data->channel, TRUE);
+ data->watch = g_io_add_watch(data->channel, G_IO_OUT | G_IO_IN,
+ socket_event, data);
+ data->timeout = g_timeout_add_seconds(ms_time, connect_timeout, data);
+
+ ret = connect(data->sock, (struct sockaddr *)remote_host,
+ sizeof(struct sockaddr));
+ if (ret < 0 && errno != EINPROGRESS) {
+ fprintf(stdout, "%d %d \n", ret, EINPROGRESS);
+ perror("Could not connect");
+ remove_timeout(data);
+ goto error;
+ }
+
+ g_free(remote_host);
+ free(ip);
+ return 0;
+
+error:
+ g_free(remote_host);
+ free(ip);
+
+ if (data->get_page)
+ data->get_page(data, NULL, 0, GET_PAGE_FAILED);
+
+ return ret;
+}
+
+static int get_page_cb(struct server_data *data, char *page, int len,
+ enum get_page_status status)
+{
+ fprintf(stdout, "\npage status %d\n", status);
+ if (status == GET_PAGE_SUCCESS && page != NULL && len > 0)
+ fprintf(stdout, "%s \n", page);
+ return 0;
+}
+
+static void usage()
+{
+ fprintf(stderr, "USAGE: protal-test <host> [page]\n");
+}
+
+int main(int argc, char **argv)
+{
+ char *host;
+ char *page = PAGE;
+ struct server_data *data;
+
+ if (argc == 1) {
+ usage();
+ exit(2);
+ }
+
+ host = argv[1];
+ if (argc > 2)
+ page = argv[2];
+
+ data = g_try_new0(struct server_data, 1);
+ if (data == NULL)
+ exit(1);
+
+ strcpy(data->host, host);
+ strcpy(data->page, page);
+ data->get_page = get_page_cb;
+ data->timeout = 0;
+
+ main_loop = g_main_loop_new(NULL, FALSE);
+
+ get_html(data, CONNECT_TIMEOUT);
+
+ g_main_loop_run(main_loop);
+
+ remove_connection(data);
+
+ return 0;
+}