Commit 2392156f authored by Sergey Lyubka's avatar Sergey Lyubka

Not exiting on a wrong option, best effort to recover

parent cf1811f8
...@@ -108,6 +108,13 @@ static void vnotify(const char *fmt, va_list ap, int must_exit) { ...@@ -108,6 +108,13 @@ static void vnotify(const char *fmt, va_list ap, int must_exit) {
} }
} }
static void notify(const char *fmt, ...) {
va_list ap;
va_start(ap, fmt);
vnotify(fmt, ap, 0);
va_end(ap);
}
static void die(const char *fmt, ...) { static void die(const char *fmt, ...) {
va_list ap; va_list ap;
va_start(ap, fmt); va_start(ap, fmt);
...@@ -263,9 +270,9 @@ static void verify_existence(char **options, const char *option_name, ...@@ -263,9 +270,9 @@ static void verify_existence(char **options, const char *option_name,
if (path != NULL && (stat(path, &st) != 0 || if (path != NULL && (stat(path, &st) != 0 ||
((S_ISDIR(st.st_mode) ? 1 : 0) != must_be_dir))) { ((S_ISDIR(st.st_mode) ? 1 : 0) != must_be_dir))) {
die("Invalid path for %s: [%s]: (%s). Make sure that path is either " notify("Invalid path for %s: [%s]: (%s). Make sure that path is either "
"absolute, or it is relative to mongoose executable.", "absolute, or it is relative to mongoose executable.",
option_name, path, strerror(errno)); option_name, path, strerror(errno));
} }
} }
...@@ -412,7 +419,9 @@ static void start_mongoose(int argc, char *argv[]) { ...@@ -412,7 +419,9 @@ static void start_mongoose(int argc, char *argv[]) {
for (i = 0; options[i] != NULL; i += 2) { for (i = 0; options[i] != NULL; i += 2) {
const char *msg = mg_set_option(server, options[i], options[i + 1]); const char *msg = mg_set_option(server, options[i], options[i + 1]);
if (msg != NULL) die("Failed to set option [%s]: %s", options[i], msg); if (msg != NULL) {
notify("Failed to set option [%s]: %s", options[i], msg);
}
free(options[i]); free(options[i]);
free(options[i + 1]); free(options[i + 1]);
} }
......
...@@ -67,6 +67,9 @@ typedef struct _stati64 file_stat_t; ...@@ -67,6 +67,9 @@ typedef struct _stati64 file_stat_t;
#ifndef EINPROGRESS #ifndef EINPROGRESS
#define EINPROGRESS WSAEINPROGRESS #define EINPROGRESS WSAEINPROGRESS
#endif #endif
#ifndef EWOULDBLOCK
#define EWOULDBLOCK WSAEWOULDBLOCK
#endif
#define mutex_init(x) InitializeCriticalSection(x) #define mutex_init(x) InitializeCriticalSection(x)
#define mutex_destroy(x) DeleteCriticalSection(x) #define mutex_destroy(x) DeleteCriticalSection(x)
#define mutex_lock(x) EnterCriticalSection(x) #define mutex_lock(x) EnterCriticalSection(x)
...@@ -788,7 +791,13 @@ static int mg_socketpair(sock_t sp[2]) { ...@@ -788,7 +791,13 @@ static int mg_socketpair(sock_t sp[2]) {
} }
static int is_error(int n) { static int is_error(int n) {
return n == 0 || (n < 0 && errno != EINTR && errno != EAGAIN); return n == 0 ||
(n < 0 && errno != EINTR && errno != EINPROGRESS &&
errno != EAGAIN && errno != EWOULDBLOCK
#ifdef _WIN32
&& WSAGetLastError() != WSAEINTR && WSAGetLastError() != WSAEWOULDBLOCK
#endif
);
} }
static void discard_leading_iobuf_bytes(struct iobuf *io, int n) { static void discard_leading_iobuf_bytes(struct iobuf *io, int n) {
...@@ -1206,11 +1215,13 @@ static void forward_post_data(struct connection *conn) { ...@@ -1206,11 +1215,13 @@ static void forward_post_data(struct connection *conn) {
static sock_t open_listening_socket(union socket_address *sa) { static sock_t open_listening_socket(union socket_address *sa) {
sock_t on = 1, sock = INVALID_SOCKET; sock_t on = 1, sock = INVALID_SOCKET;
if ((sock = socket(sa->sa.sa_family, SOCK_STREAM, 6)) == INVALID_SOCKET || if ((sock = socket(sa->sa.sa_family, SOCK_STREAM, 6)) != INVALID_SOCKET &&
setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, (void *) &on, sizeof(on)) || !setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, (void *) &on, sizeof(on)) &&
bind(sock, &sa->sa, sa->sa.sa_family == AF_INET ? !bind(sock, &sa->sa, sa->sa.sa_family == AF_INET ?
sizeof(sa->sin) : sizeof(sa->sa)) != 0 || sizeof(sa->sin) : sizeof(sa->sa)) &&
listen(sock, SOMAXCONN) != 0) { !listen(sock, SOMAXCONN)) {
set_non_blocking_mode(sock);
} else if (sock != INVALID_SOCKET) {
closesocket(sock); closesocket(sock);
sock = INVALID_SOCKET; sock = INVALID_SOCKET;
} }
...@@ -1925,28 +1936,27 @@ static void call_uri_handler(struct connection *conn) { ...@@ -1925,28 +1936,27 @@ static void call_uri_handler(struct connection *conn) {
} }
static void callback_http_client_on_connect(struct connection *conn) { static void callback_http_client_on_connect(struct connection *conn) {
int ok = 1; int ok = 1, ret;
socklen_t len = sizeof(ok); socklen_t len = sizeof(ok);
conn->flags &= ~CONN_CONNECTING; conn->flags &= ~CONN_CONNECTING;
if (getsockopt(conn->client_sock, SOL_SOCKET, SO_ERROR, (char *) &ok, ret = getsockopt(conn->client_sock, SOL_SOCKET, SO_ERROR, (char *) &ok, &len);
&len) == 0 && ok == 0) {
#ifdef MONGOOSE_USE_SSL #ifdef MONGOOSE_USE_SSL
if (conn->ssl != NULL) { if (ret == 0 && ok == 0 && conn->ssl != NULL) {
int res = SSL_connect(conn->ssl), ssl_err = SSL_get_error(conn->ssl, res); int res = SSL_connect(conn->ssl), ssl_err = SSL_get_error(conn->ssl, res);
//DBG(("%p res %d %d", conn, res, ssl_err)); //DBG(("%p res %d %d", conn, res, ssl_err));
if (res == 1) { if (res == 1) {
conn->flags = CONN_SSL_HANDS_SHAKEN; conn->flags = CONN_SSL_HANDS_SHAKEN;
} else if (res == 0 || ssl_err == 2 || ssl_err == 3) { } else if (res == 0 || ssl_err == 2 || ssl_err == 3) {
conn->flags |= CONN_CONNECTING; conn->flags |= CONN_CONNECTING;
return; // Call us again return; // Call us again
} else { } else {
ok = 1; ok = 1;
}
} }
#endif
} }
conn->mg_conn.status_code = ok == 0 ? MG_CONNECT_SUCCESS : MG_CONNECT_FAILURE; #endif
conn->mg_conn.status_code =
(ret == 0 && ok == 0) ? MG_CONNECT_SUCCESS : MG_CONNECT_FAILURE;
if (conn->handler(&conn->mg_conn) || ok != 0) { if (conn->handler(&conn->mg_conn) || ok != 0) {
conn->flags |= CONN_CLOSE; conn->flags |= CONN_CLOSE;
} }
...@@ -3608,7 +3618,7 @@ int mg_connect(struct mg_server *server, const char *host, int port, ...@@ -3608,7 +3618,7 @@ int mg_connect(struct mg_server *server, const char *host, int port,
set_non_blocking_mode(sock); set_non_blocking_mode(sock);
connect_ret_val = connect(sock, (struct sockaddr *) &sin, sizeof(sin)); connect_ret_val = connect(sock, (struct sockaddr *) &sin, sizeof(sin));
if (connect_ret_val != 0 && errno != EINPROGRESS) { if (is_error(connect_ret_val)) {
return 0; return 0;
} else if ((conn = (struct connection *) calloc(1, sizeof(*conn))) == NULL) { } else if ((conn = (struct connection *) calloc(1, sizeof(*conn))) == NULL) {
closesocket(sock); closesocket(sock);
...@@ -3631,16 +3641,6 @@ int mg_connect(struct mg_server *server, const char *host, int port, ...@@ -3631,16 +3641,6 @@ int mg_connect(struct mg_server *server, const char *host, int port,
LINKED_LIST_ADD_TO_FRONT(&server->active_connections, &conn->link); LINKED_LIST_ADD_TO_FRONT(&server->active_connections, &conn->link);
DBG(("%p %s:%d", conn, host, port)); DBG(("%p %s:%d", conn, host, port));
if (connect_ret_val == 0) {
callback_http_client_on_connect(conn);
#if 0
conn->mg_conn.status_code = MG_CONNECT_SUCCESS;
conn->flags &= ~CONN_CONNECTING;
conn->mg_conn.content = conn->local_iobuf.buf;
handler(&conn->mg_conn);
#endif
}
return 1; return 1;
} }
...@@ -3890,7 +3890,7 @@ void mg_add_uri_handler(struct mg_server *server, const char *uri, ...@@ -3890,7 +3890,7 @@ void mg_add_uri_handler(struct mg_server *server, const char *uri,
mg_handler_t handler) { mg_handler_t handler) {
struct uri_handler *p = (struct uri_handler *) malloc(sizeof(*p)); struct uri_handler *p = (struct uri_handler *) malloc(sizeof(*p));
if (p != NULL) { if (p != NULL) {
LINKED_LIST_ADD_TO_FRONT(&server->uri_handlers, &p->link); LINKED_LIST_ADD_TO_TAIL(&server->uri_handlers, &p->link);
p->uri = mg_strdup(uri); p->uri = mg_strdup(uri);
p->handler = handler; p->handler = handler;
} }
...@@ -4082,8 +4082,6 @@ const char *mg_set_option(struct mg_server *server, const char *name, ...@@ -4082,8 +4082,6 @@ const char *mg_set_option(struct mg_server *server, const char *name,
server->listening_sock = open_listening_socket(&server->lsa); server->listening_sock = open_listening_socket(&server->lsa);
if (server->listening_sock == INVALID_SOCKET) { if (server->listening_sock == INVALID_SOCKET) {
error_msg = "Cannot bind to port"; error_msg = "Cannot bind to port";
} else {
set_non_blocking_mode(server->listening_sock);
} }
#ifndef _WIN32 #ifndef _WIN32
} else if (ind == RUN_AS_USER) { } else if (ind == RUN_AS_USER) {
......
...@@ -475,6 +475,8 @@ static int cb4(struct mg_connection *conn) { ...@@ -475,6 +475,8 @@ static int cb4(struct mg_connection *conn) {
} }
static int cb3(struct mg_connection *conn) { static int cb3(struct mg_connection *conn) {
printf("cb3: %d\n", conn->status_code);
fflush(stdout);
sprintf((char *) conn->connection_param, "%d", conn->status_code); sprintf((char *) conn->connection_param, "%d", conn->status_code);
return 1; return 1;
} }
...@@ -488,7 +490,7 @@ static const char *test_mg_connect(void) { ...@@ -488,7 +490,7 @@ static const char *test_mg_connect(void) {
mg_add_uri_handler(server, "/x", cb4h); mg_add_uri_handler(server, "/x", cb4h);
ASSERT(mg_connect(server, "", 0, 0, NULL, NULL) == 0); ASSERT(mg_connect(server, "", 0, 0, NULL, NULL) == 0);
ASSERT(mg_connect(server, "127.0.0.1", atoi(HTTP_PORT), 0, cb2, buf2) == 1); ASSERT(mg_connect(server, "127.0.0.1", atoi(HTTP_PORT), 0, cb2, buf2) == 1);
ASSERT(mg_connect(server, "127.0.0.1", 1, 0, cb3, buf3) == 1); ASSERT(mg_connect(server, "127.0.0.1", 29, 0, cb3, buf3) == 1);
ASSERT(mg_connect(server, "127.0.0.1", atoi(HTTP_PORT), 0, cb4, buf4) == 1); ASSERT(mg_connect(server, "127.0.0.1", atoi(HTTP_PORT), 0, cb4, buf4) == 1);
{ int i; for (i = 0; i < 50; i++) mg_poll_server(server, 0); } { int i; for (i = 0; i < 50; i++) mg_poll_server(server, 0); }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment