| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388 |
- /*
- * Copyright (C) 2011, 2012, 2013, 2014 Citrix Systems
- *
- * All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. Neither the name of the project nor the names of its contributors
- * may be used to endorse or promote products derived from this software
- * without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND
- * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
- * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE PROJECT OR CONTRIBUTORS BE LIABLE
- * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- */
- #include "ns_ioalib_impl.h"
- #include "http_server.h"
- #include <event2/http.h>
- #include <event2/keyvalq_struct.h>
- #include <time.h>
- //////////////////////////////////////
- struct headers_list {
- size_t n;
- char **keys;
- char **values;
- };
- struct http_headers {
- struct evkeyvalq *uri_headers;
- struct headers_list *post_headers;
- };
- //////////////////////////////////////
- static void write_http_echo(ioa_socket_handle s)
- {
- if(s && !ioa_socket_tobeclosed(s)) {
- SOCKET_APP_TYPE sat = get_ioa_socket_app_type(s);
- if((sat == HTTP_CLIENT_SOCKET) || (sat == HTTPS_CLIENT_SOCKET)) {
- ioa_network_buffer_handle nbh_http = ioa_network_buffer_allocate(s->e);
- size_t len_http = ioa_network_buffer_get_size(nbh_http);
- uint8_t *data = ioa_network_buffer_data(nbh_http);
- char data_http[1025];
- char content_http[1025];
- const char* title = "TURN Server";
- snprintf(content_http,sizeof(content_http)-1,"<!DOCTYPE html>\r\n<html>\r\n <head>\r\n <title>%s</title>\r\n </head>\r\n <body>\r\n <b>%s</b> <br> <b><i>use https connection for the admin session</i></b>\r\n </body>\r\n</html>\r\n",title,title);
- snprintf(data_http,sizeof(data_http)-1,"HTTP/1.0 200 OK\r\nServer: %s\r\nContent-Type: text/html; charset=UTF-8\r\nContent-Length: %d\r\n\r\n%.906s",TURN_SOFTWARE,(int)strlen(content_http),content_http);
- len_http = strlen(data_http);
- bcopy(data_http,data,len_http);
- ioa_network_buffer_set_size(nbh_http,len_http);
- send_data_from_ioa_socket_nbh(s, NULL, nbh_http, TTL_IGNORE, TOS_IGNORE,NULL);
- }
- }
- }
- void handle_http_echo(ioa_socket_handle s) {
- write_http_echo(s);
- }
- const char* get_http_date_header()
- {
- static char buffer_date[256];
- static char buffer_header[1025];
- static const char* wds[]={"Sun","Mon","Tue","Wed","Thu","Fri","Sat"};
- static const char* mons[]={"Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"};
- time_t now = time(NULL);
- struct tm *gmtm = gmtime(&now);
- buffer_header[0]=0;
- buffer_date[0]=0;
- if(gmtm) {
- snprintf(buffer_date,sizeof(buffer_date)-1,"%s, %d %s %d %d:%d:%d GMT",wds[gmtm->tm_wday], gmtm->tm_mday, mons[gmtm->tm_mon], gmtm->tm_year+1900, gmtm->tm_hour, gmtm->tm_min, gmtm->tm_sec);
- buffer_date[sizeof(buffer_date)-1]=0;
- snprintf(buffer_header,sizeof(buffer_header)-1,"Date: %s\r\n",buffer_date);
- buffer_header[sizeof(buffer_header)-1]=0;
- }
- return buffer_header;
- }
- ///////////////////////////////////////////////
- static struct headers_list * post_parse(char *data, size_t data_len)
- {
- while((*data=='\r')||(*data=='\n')) { ++data; --data_len; }
- if (data_len) {
- char *post_data = (char*)calloc(data_len + 1, sizeof(char));
- if (post_data != NULL) {
- memcpy(post_data, data, data_len);
- char *fmarker = NULL;
- char *fsplit = strtok_r(post_data, "&", &fmarker);
- struct headers_list *list = (struct headers_list*)malloc(sizeof(struct headers_list));
- bzero(list,sizeof(struct headers_list));
- while (fsplit != NULL) {
- char *vmarker = NULL;
- char *key = strtok_r(fsplit, "=", &vmarker);
- if (key == NULL)
- break;
- else {
- char *value = strtok_r(NULL, "=", &vmarker);
- char empty[1];
- empty[0]=0;
- value = value ? value : empty;
- value = evhttp_decode_uri(value);
- char *p = value;
- while (*p) {
- if (*p == '+')
- *p = ' ';
- p++;
- }
- list->keys = (char**)realloc(list->keys,sizeof(char*)*(list->n+1));
- list->keys[list->n] = strdup(key);
- list->values = (char**)realloc(list->values,sizeof(char*)*(list->n+1));
- list->values[list->n] = value;
- ++(list->n);
- fsplit = strtok_r(NULL, "&", &fmarker);
- }
- }
- free(post_data);
- return list;
- }
- }
- return NULL;
- }
- static struct http_request* parse_http_request_1(struct http_request* ret, char* request, int parse_post)
- {
- if(ret && request) {
- char* s = strstr(request," HTTP/");
- if(!s) {
- free(ret);
- ret = NULL;
- } else {
- *s = 0;
- struct evhttp_uri *uri = evhttp_uri_parse(request);
- if(!uri) {
- free(ret);
- ret = NULL;
- } else {
- const char *query = evhttp_uri_get_query(uri);
- if(query) {
- struct evkeyvalq* kv = (struct evkeyvalq*)malloc(sizeof(struct evkeyvalq));
- bzero(kv,sizeof(struct evkeyvalq));
- if(evhttp_parse_query_str(query, kv)<0) {
- free(ret);
- ret = NULL;
- } else {
- ret->headers = (struct http_headers*)malloc(sizeof(struct http_headers));
- bzero(ret->headers,sizeof(struct http_headers));
- ret->headers->uri_headers = kv;
- }
- }
- const char *path = evhttp_uri_get_path(uri);
- if(path && ret)
- ret->path = strdup(path);
- evhttp_uri_free(uri);
- if(parse_post && ret) {
- char *body = strstr(s+1,"\r\n\r\n");
- if(body && body[0]) {
- if(!ret->headers) {
- ret->headers = (struct http_headers*)malloc(sizeof(struct http_headers));
- bzero(ret->headers,sizeof(struct http_headers));
- }
- ret->headers->post_headers = post_parse(body,strlen(body));
- }
- }
- }
- *s = ' ';
- }
- }
- return ret;
- }
- struct http_request* parse_http_request(char* request) {
- struct http_request* ret = NULL;
- if(request) {
- ret = (struct http_request*)malloc(sizeof(struct http_request));
- bzero(ret,sizeof(struct http_request));
- if(strstr(request,"GET ") == request) {
- ret->rtype = HRT_GET;
- ret = parse_http_request_1(ret,request+4,0);
- } else if(strstr(request,"HEAD ") == request) {
- ret->rtype = HRT_HEAD;
- ret = parse_http_request_1(ret,request+5,0);
- } else if(strstr(request,"POST ") == request) {
- ret->rtype = HRT_POST;
- ret = parse_http_request_1(ret,request+5,1);
- } else if(strstr(request,"PUT ") == request) {
- ret->rtype = HRT_PUT;
- ret = parse_http_request_1(ret,request+4,1);
- } else if(strstr(request,"DELETE ") == request) {
- ret->rtype = HRT_DELETE;
- ret = parse_http_request_1(ret,request+7,1);
- } else {
- free(ret);
- ret = NULL;
- }
- }
- return ret;
- }
- static const char * get_headers_list_value(struct headers_list *h, const char* key) {
- const char* ret = NULL;
- if(h && h->keys && h->values && key && key[0]) {
- size_t i = 0;
- for(i=0;i<h->n;++i) {
- if(h->keys[i] && !strcmp(key,h->keys[i]) && h->values[i]) {
- ret = h->values[i];
- break;
- }
- }
- }
- return ret;
- }
- static void free_headers_list(struct headers_list *h) {
- if(h) {
- if(h->keys) {
- size_t i = 0;
- for(i=0;i<h->n;++i) {
- if(h->keys[i]) {
- free(h->keys[i]);
- h->keys[i]=NULL;
- }
- }
- free(h->keys);
- h->keys = NULL;
- }
- if(h->values) {
- size_t i = 0;
- for(i=0;i<h->n;++i) {
- if(h->values[i]) {
- free(h->values[i]);
- h->values[i]=NULL;
- }
- }
- free(h->values);
- h->values = NULL;
- }
- h->n = 0;
- free(h);
- }
- }
- const char *get_http_header_value(const struct http_request *request, const char* key, const char* default_value) {
- const char *ret = NULL;
- if(key && key[0] && request && request->headers) {
- if(request->headers->uri_headers) {
- ret = evhttp_find_header(request->headers->uri_headers,key);
- }
- if(!ret && request->headers->post_headers) {
- ret = get_headers_list_value(request->headers->post_headers,key);
- }
- }
- if(!ret) {
- ret = default_value;
- }
- return ret;
- }
- void free_http_request(struct http_request *request) {
- if(request) {
- if(request->path) {
- free(request->path);
- request->path = NULL;
- }
- if(request->headers) {
- if(request->headers->uri_headers) {
- evhttp_clear_headers(request->headers->uri_headers);
- free(request->headers->uri_headers);
- request->headers->uri_headers = NULL;
- }
- if(request->headers->post_headers) {
- free_headers_list(request->headers->post_headers);
- request->headers->post_headers = NULL;
- }
- free(request->headers);
- request->headers = NULL;
- }
- free(request);
- }
- }
- ////////////////////////////////////////////
- struct str_buffer {
- size_t capacity;
- size_t sz;
- char* buffer;
- };
- struct str_buffer* str_buffer_new(void)
- {
- struct str_buffer* ret = (struct str_buffer*)malloc(sizeof(struct str_buffer));
- bzero(ret,sizeof(struct str_buffer));
- ret->buffer = (char*)malloc(1);
- ret->buffer[0] = 0;
- ret->capacity = 1;
- return ret;
- }
- void str_buffer_append(struct str_buffer* sb, const char* str)
- {
- if(sb && str && str[0]) {
- size_t len = strlen(str);
- while(sb->sz + len + 1 > sb->capacity) {
- sb->capacity += len + 1024;
- sb->buffer = (char*)realloc(sb->buffer,sb->capacity);
- }
- bcopy(str,sb->buffer+sb->sz,len+1);
- sb->sz += len;
- }
- }
- void str_buffer_append_sz(struct str_buffer* sb, size_t sz)
- {
- char ssz[129];
- snprintf(ssz,sizeof(ssz)-1,"%lu",(unsigned long)sz);
- str_buffer_append(sb,ssz);
- }
- void str_buffer_append_sid(struct str_buffer* sb, turnsession_id sid)
- {
- char ssz[129];
- snprintf(ssz,sizeof(ssz)-1,"%018llu",(unsigned long long)sid);
- str_buffer_append(sb,ssz);
- }
- const char* str_buffer_get_str(const struct str_buffer *sb)
- {
- if(sb) {
- return sb->buffer;
- }
- return NULL;
- }
- size_t str_buffer_get_str_len(const struct str_buffer *sb)
- {
- if(sb) {
- return sb->sz;
- }
- return 0;
- }
- void str_buffer_free(struct str_buffer *sb)
- {
- if(sb) {
- free(sb->buffer);
- free(sb);
- }
- }
- ///////////////////////////////////////////////
|