Commit 51ad50a6 authored by Deomid Ryabkov's avatar Deomid Ryabkov Committed by rojer

Rewrite URI -> path mapping, Windows fixes

    PUBLISHED_FROM=6088958e92af2b028646816435892828ce9f7625
parent 148e1926
#include "mongoose.h" #include "mongoose.h"
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/internal.h" #line 0 "./src/internal.h"
#endif #endif
/* /*
...@@ -84,14 +84,15 @@ MG_INTERNAL size_t recv_avail_size(struct mg_connection *conn, size_t max); ...@@ -84,14 +84,15 @@ MG_INTERNAL size_t recv_avail_size(struct mg_connection *conn, size_t max);
MG_INTERNAL struct mg_connection *mg_create_connection( MG_INTERNAL struct mg_connection *mg_create_connection(
struct mg_mgr *mgr, mg_event_handler_t callback, struct mg_mgr *mgr, mg_event_handler_t callback,
struct mg_add_sock_opts opts); struct mg_add_sock_opts opts);
#ifndef MG_DISABLE_FILESYSTEM #ifndef MG_DISABLE_FILESYSTEM
MG_INTERNAL int find_index_file(char *path, size_t path_len, const char *list, MG_INTERNAL int mg_uri_to_local_path(struct http_message *hm,
cs_stat_t *stp); const struct mg_serve_http_opts *opts,
char **local_path,
struct mg_str *remainder);
#endif #endif
#ifdef _WIN32 #ifdef _WIN32
void to_wchar(const char *path, wchar_t *wbuf, size_t wbuf_len); /* Retur value is the same as for MultiByteToWideChar. */
int to_wchar(const char *path, wchar_t *wbuf, size_t wbuf_len);
#endif #endif
/* /*
...@@ -124,7 +125,7 @@ extern void *(*test_malloc)(size_t size); ...@@ -124,7 +125,7 @@ extern void *(*test_malloc)(size_t size);
extern void *(*test_calloc)(size_t count, size_t size); extern void *(*test_calloc)(size_t count, size_t size);
#endif /* MG_INTERNAL_HEADER_INCLUDED */ #endif /* MG_INTERNAL_HEADER_INCLUDED */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/../../common/base64.c" #line 0 "./src/../../common/base64.c"
#endif #endif
/* /*
...@@ -325,7 +326,7 @@ int cs_base64_decode(const unsigned char *s, int len, char *dst) { ...@@ -325,7 +326,7 @@ int cs_base64_decode(const unsigned char *s, int len, char *dst) {
} }
#endif /* EXCLUDE_COMMON */ #endif /* EXCLUDE_COMMON */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/../../common/cs_dbg.c" #line 0 "./src/../../common/cs_dbg.c"
#endif #endif
/* /*
...@@ -357,7 +358,7 @@ void cs_log_printf(const char *fmt, ...) { ...@@ -357,7 +358,7 @@ void cs_log_printf(const char *fmt, ...) {
void cs_log_set_level(enum cs_log_level level) { void cs_log_set_level(enum cs_log_level level) {
s_cs_log_level = level; s_cs_log_level = level;
} }
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/../../common/cs_dirent.c" #line 0 "./src/../../common/cs_dirent.c"
#endif #endif
/* /*
...@@ -425,7 +426,7 @@ int closedir(DIR *dir) { ...@@ -425,7 +426,7 @@ int closedir(DIR *dir) {
} }
struct dirent *readdir(DIR *dir) { struct dirent *readdir(DIR *dir) {
struct dirent *result = 0; struct dirent *result = NULL;
if (dir) { if (dir) {
if (dir->handle != INVALID_HANDLE_VALUE) { if (dir->handle != INVALID_HANDLE_VALUE) {
...@@ -493,7 +494,7 @@ int mkdir(const char *path, mode_t mode) { ...@@ -493,7 +494,7 @@ int mkdir(const char *path, mode_t mode) {
#endif /* CS_ENABLE_SPIFFS */ #endif /* CS_ENABLE_SPIFFS */
#endif /* EXCLUDE_COMMON */ #endif /* EXCLUDE_COMMON */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/../../common/cs_time.c" #line 0 "./src/../../common/cs_time.c"
#endif #endif
/* /*
...@@ -521,7 +522,7 @@ double cs_time() { ...@@ -521,7 +522,7 @@ double cs_time() {
#endif #endif
return now; return now;
} }
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/../deps/frozen/frozen.c" #line 0 "./src/../deps/frozen/frozen.c"
#endif #endif
/* /*
...@@ -1021,7 +1022,7 @@ int json_emit(char *buf, int buf_len, const char *fmt, ...) { ...@@ -1021,7 +1022,7 @@ int json_emit(char *buf, int buf_len, const char *fmt, ...) {
return len; return len;
} }
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/../../common/md5.c" #line 0 "./src/../../common/md5.c"
#endif #endif
/* /*
...@@ -1268,7 +1269,7 @@ char *cs_md5(char buf[33], ...) { ...@@ -1268,7 +1269,7 @@ char *cs_md5(char buf[33], ...) {
} }
#endif /* EXCLUDE_COMMON */ #endif /* EXCLUDE_COMMON */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/../../common/mbuf.c" #line 0 "./src/../../common/mbuf.c"
#endif #endif
/* /*
...@@ -1365,7 +1366,7 @@ void mbuf_remove(struct mbuf *mb, size_t n) { ...@@ -1365,7 +1366,7 @@ void mbuf_remove(struct mbuf *mb, size_t n) {
} }
#endif /* EXCLUDE_COMMON */ #endif /* EXCLUDE_COMMON */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/../../common/sha1.c" #line 0 "./src/../../common/sha1.c"
#endif #endif
/* Copyright(c) By Steve Reid <steve@edmweb.com> */ /* Copyright(c) By Steve Reid <steve@edmweb.com> */
...@@ -1619,7 +1620,7 @@ void cs_hmac_sha1(const unsigned char *key, size_t keylen, ...@@ -1619,7 +1620,7 @@ void cs_hmac_sha1(const unsigned char *key, size_t keylen,
} }
#endif /* EXCLUDE_COMMON */ #endif /* EXCLUDE_COMMON */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/../../common/str_util.c" #line 0 "./src/../../common/str_util.c"
#endif #endif
/* /*
...@@ -1842,7 +1843,8 @@ int c_snprintf(char *buf, size_t buf_size, const char *fmt, ...) { ...@@ -1842,7 +1843,8 @@ int c_snprintf(char *buf, size_t buf_size, const char *fmt, ...) {
} }
#ifdef _WIN32 #ifdef _WIN32
void to_wchar(const char *path, wchar_t *wbuf, size_t wbuf_len) { int to_wchar(const char *path, wchar_t *wbuf, size_t wbuf_len) {
int ret;
char buf[MAX_PATH * 2], buf2[MAX_PATH * 2], *p; char buf[MAX_PATH * 2], buf2[MAX_PATH * 2], *p;
strncpy(buf, path, sizeof(buf)); strncpy(buf, path, sizeof(buf));
...@@ -1852,22 +1854,26 @@ void to_wchar(const char *path, wchar_t *wbuf, size_t wbuf_len) { ...@@ -1852,22 +1854,26 @@ void to_wchar(const char *path, wchar_t *wbuf, size_t wbuf_len) {
p = buf + strlen(buf) - 1; p = buf + strlen(buf) - 1;
while (p > buf && p[-1] != ':' && (p[0] == '\\' || p[0] == '/')) *p-- = '\0'; while (p > buf && p[-1] != ':' && (p[0] == '\\' || p[0] == '/')) *p-- = '\0';
memset(wbuf, 0, wbuf_len * sizeof(wchar_t));
ret = MultiByteToWideChar(CP_UTF8, 0, buf, -1, wbuf, (int) wbuf_len);
/* /*
* Convert to Unicode and back. If doubly-converted string does not * Convert back to Unicode. If doubly-converted string does not match the
* match the original, something is fishy, reject. * original, something is fishy, reject.
*/ */
memset(wbuf, 0, wbuf_len * sizeof(wchar_t));
MultiByteToWideChar(CP_UTF8, 0, buf, -1, wbuf, (int) wbuf_len);
WideCharToMultiByte(CP_UTF8, 0, wbuf, (int) wbuf_len, buf2, sizeof(buf2), WideCharToMultiByte(CP_UTF8, 0, wbuf, (int) wbuf_len, buf2, sizeof(buf2),
NULL, NULL); NULL, NULL);
if (strcmp(buf, buf2) != 0) { if (strcmp(buf, buf2) != 0) {
wbuf[0] = L'\0'; wbuf[0] = L'\0';
ret = 0;
} }
return ret;
} }
#endif /* _WIN32 */ #endif /* _WIN32 */
#endif /* EXCLUDE_COMMON */ #endif /* EXCLUDE_COMMON */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/net.c" #line 0 "./src/net.c"
#endif #endif
/* /*
...@@ -2818,6 +2824,7 @@ int mg_check_ip_acl(const char *acl, uint32_t remote_ip) { ...@@ -2818,6 +2824,7 @@ int mg_check_ip_acl(const char *acl, uint32_t remote_ip) {
} }
} }
DBG(("%08x %c", remote_ip, allowed));
return allowed == '+'; return allowed == '+';
} }
...@@ -2846,7 +2853,7 @@ double mg_set_timer(struct mg_connection *c, double timestamp) { ...@@ -2846,7 +2853,7 @@ double mg_set_timer(struct mg_connection *c, double timestamp) {
double mg_time() { double mg_time() {
return cs_time(); return cs_time();
} }
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/net_if_socket.c" #line 0 "./src/net_if_socket.c"
#endif #endif
/* /*
...@@ -2927,7 +2934,9 @@ void mg_if_connect_udp(struct mg_connection *nc) { ...@@ -2927,7 +2934,9 @@ void mg_if_connect_udp(struct mg_connection *nc) {
int mg_if_listen_tcp(struct mg_connection *nc, union socket_address *sa) { int mg_if_listen_tcp(struct mg_connection *nc, union socket_address *sa) {
sock_t sock = mg_open_listening_socket(sa, SOCK_STREAM); sock_t sock = mg_open_listening_socket(sa, SOCK_STREAM);
if (sock < 0) return (errno ? errno : 1); if (sock == INVALID_SOCKET) {
return (errno ? errno : 1);
}
mg_sock_set(nc, sock); mg_sock_set(nc, sock);
return 0; return 0;
} }
...@@ -3197,7 +3206,7 @@ static int mg_ssl_err(struct mg_connection *conn, int res) { ...@@ -3197,7 +3206,7 @@ static int mg_ssl_err(struct mg_connection *conn, int res) {
static void mg_ssl_begin(struct mg_connection *nc) { static void mg_ssl_begin(struct mg_connection *nc) {
int server_side = nc->listener != NULL; int server_side = nc->listener != NULL;
int res = server_side ? SSL_accept(nc->ssl) : SSL_connect(nc->ssl); int res = server_side ? SSL_accept(nc->ssl) : SSL_connect(nc->ssl);
DBG(("%p %d res %d %d %d", nc, server_side, res, errno, mg_ssl_err(nc, res))); DBG(("%p %d res %d %d", nc, server_side, res, errno));
if (res == 1) { if (res == 1) {
nc->flags |= MG_F_SSL_HANDSHAKE_DONE; nc->flags |= MG_F_SSL_HANDSHAKE_DONE;
...@@ -3650,16 +3659,14 @@ int mg_socketpair(sock_t sp[2], int sock_type) { ...@@ -3650,16 +3659,14 @@ int mg_socketpair(sock_t sp[2], int sock_type) {
static void mg_sock_get_addr(sock_t sock, int remote, static void mg_sock_get_addr(sock_t sock, int remote,
union socket_address *sa) { union socket_address *sa) {
socklen_t slen = sizeof(*sa);
memset(sa, 0, slen);
#ifndef MG_CC3200 #ifndef MG_CC3200
socklen_t slen = sizeof(sa);
memset(sa, 0, sizeof(*sa));
if (remote) { if (remote) {
getpeername(sock, &sa->sa, &slen); getpeername(sock, &sa->sa, &slen);
} else { } else {
getsockname(sock, &sa->sa, &slen); getsockname(sock, &sa->sa, &slen);
} }
#else
memset(sa, 0, sizeof(*sa));
#endif #endif
} }
...@@ -3675,7 +3682,7 @@ void mg_if_get_conn_addr(struct mg_connection *nc, int remote, ...@@ -3675,7 +3682,7 @@ void mg_if_get_conn_addr(struct mg_connection *nc, int remote,
} }
#endif /* !MG_DISABLE_SOCKET_IF */ #endif /* !MG_DISABLE_SOCKET_IF */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/multithreading.c" #line 0 "./src/multithreading.c"
#endif #endif
/* /*
...@@ -3782,7 +3789,7 @@ void mg_enable_multithreading(struct mg_connection *nc) { ...@@ -3782,7 +3789,7 @@ void mg_enable_multithreading(struct mg_connection *nc) {
nc->handler = multithreaded_ev_handler; nc->handler = multithreaded_ev_handler;
} }
#endif #endif
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/uri.c" #line 0 "./src/uri.c"
#endif #endif
/* /*
...@@ -3807,7 +3814,7 @@ static void parse_uri_component(const char **p, const char *end, char sep, ...@@ -3807,7 +3814,7 @@ static void parse_uri_component(const char **p, const char *end, char sep,
} }
} }
res->len = (*p) - res->p; res->len = (*p) - res->p;
(*p)++; if (*p < end) (*p)++;
} }
int mg_parse_uri(struct mg_str uri, struct mg_str *scheme, int mg_parse_uri(struct mg_str uri, struct mg_str *scheme,
...@@ -3924,7 +3931,42 @@ int mg_parse_uri(struct mg_str uri, struct mg_str *scheme, ...@@ -3924,7 +3931,42 @@ int mg_parse_uri(struct mg_str uri, struct mg_str *scheme,
return 0; return 0;
} }
#ifdef NS_MODULE_LINES
/* Normalize the URI path. Remove/resolve "." and "..". */
int mg_normalize_uri_path(const struct mg_str *in, struct mg_str *out) {
const char *s = in->p, *se = s + in->len;
char *cp = (char *) out->p, *d;
if (in->len == 0 || *s != '/') {
out->len = 0;
return 0;
}
d = cp;
while (s < se) {
const char *next = s;
struct mg_str component;
parse_uri_component(&next, se, '/', &component);
if (mg_vcmp(&component, ".") == 0) {
/* Yum. */
} else if (mg_vcmp(&component, "..") == 0) {
/* Backtrack to previous slash. */
if (d > cp + 1 && *(d - 1) == '/') d--;
while (d > cp && *(d - 1) != '/') d--;
} else {
memmove(d, s, next - s);
d += next - s;
}
s = next;
}
if (d == cp) *d++ = '/';
out->p = cp;
out->len = d - cp;
return 1;
}
#ifdef MG_MODULE_LINES
#line 0 "./src/http.c" #line 0 "./src/http.c"
#endif #endif
/* /*
...@@ -4828,6 +4870,15 @@ void mg_send_response_line(struct mg_connection *nc, int status_code, ...@@ -4828,6 +4870,15 @@ void mg_send_response_line(struct mg_connection *nc, int status_code,
case 302: case 302:
status_message = "Found"; status_message = "Found";
break; break;
case 401:
status_message = "Unauthorized";
break;
case 403:
status_message = "Forbidden";
break;
case 404:
status_message = "Not Found";
break;
case 416: case 416:
status_message = "Requested range not satisfiable"; status_message = "Requested range not satisfiable";
break; break;
...@@ -4898,7 +4949,7 @@ static void do_ssi_include(struct mg_connection *nc, const char *ssi, char *tag, ...@@ -4898,7 +4949,7 @@ static void do_ssi_include(struct mg_connection *nc, const char *ssi, char *tag,
sscanf(tag, " \"%[^\"]\"", file_name) == 1) { sscanf(tag, " \"%[^\"]\"", file_name) == 1) {
/* File name is relative to the currect document */ /* File name is relative to the currect document */
snprintf(path, sizeof(path), "%s", ssi); snprintf(path, sizeof(path), "%s", ssi);
if ((p = strrchr(path, '/')) != NULL) { if ((p = strrchr(path, DIRSEP)) != NULL) {
p[1] = '\0'; p[1] = '\0';
} }
snprintf(path + strlen(path), sizeof(path) - strlen(path), "%s", file_name); snprintf(path + strlen(path), sizeof(path) - strlen(path), "%s", file_name);
...@@ -5024,7 +5075,7 @@ static void handle_ssi_request(struct mg_connection *nc, const char *path, ...@@ -5024,7 +5075,7 @@ static void handle_ssi_request(struct mg_connection *nc, const char *path,
struct mg_str mime_type; struct mg_str mime_type;
if ((fp = fopen(path, "rb")) == NULL) { if ((fp = fopen(path, "rb")) == NULL) {
send_http_error(nc, 404, "Not Found"); send_http_error(nc, 404, NULL);
} else { } else {
mg_set_close_on_exec(fileno(fp)); mg_set_close_on_exec(fileno(fp));
...@@ -5078,13 +5129,25 @@ static void mg_send_http_file2(struct mg_connection *nc, const char *path, ...@@ -5078,13 +5129,25 @@ static void mg_send_http_file2(struct mg_connection *nc, const char *path,
struct proto_data_http *dp; struct proto_data_http *dp;
struct mg_str mime_type; struct mg_str mime_type;
DBG(("%p [%s]", nc, path));
free_http_proto_data(nc); free_http_proto_data(nc);
if ((dp = (struct proto_data_http *) MG_CALLOC(1, sizeof(*dp))) == NULL) { if ((dp = (struct proto_data_http *) MG_CALLOC(1, sizeof(*dp))) == NULL) {
send_http_error(nc, 500, "Server Error"); /* LCOV_EXCL_LINE */ send_http_error(nc, 500, "Server Error"); /* LCOV_EXCL_LINE */
} else if ((dp->fp = fopen(path, "rb")) == NULL) { } else if ((dp->fp = fopen(path, "rb")) == NULL) {
int code;
switch (errno) {
case EACCES:
code = 403;
break;
case ENOENT:
code = 404;
break;
default:
code = 500;
};
MG_FREE(dp); MG_FREE(dp);
nc->proto_data = NULL; nc->proto_data = NULL;
send_http_error(nc, 500, "Server Error"); send_http_error(nc, code, "Open failed");
} else if (mg_match_prefix(opts->ssi_pattern, strlen(opts->ssi_pattern), } else if (mg_match_prefix(opts->ssi_pattern, strlen(opts->ssi_pattern),
path) > 0) { path) > 0) {
nc->proto_data = (void *) dp; nc->proto_data = (void *) dp;
...@@ -5154,26 +5217,6 @@ static void mg_send_http_file2(struct mg_connection *nc, const char *path, ...@@ -5154,26 +5217,6 @@ static void mg_send_http_file2(struct mg_connection *nc, const char *path,
} }
} }
static void remove_double_dots(char *s) {
char *p = s;
while (*s != '\0') {
*p++ = *s++;
if (s[-1] == '/' || s[-1] == '\\') {
while (s[0] != '\0') {
if (s[0] == '/' || s[0] == '\\') {
s++;
} else if (s[0] == '.' && s[1] == '.') {
s += 2;
} else {
break;
}
}
}
}
*p = '\0';
}
#endif #endif
int mg_url_decode(const char *src, int src_len, char *dst, int dst_len, int mg_url_decode(const char *src, int src_len, char *dst, int dst_len,
...@@ -5337,7 +5380,8 @@ int mg_http_parse_header(struct mg_str *hdr, const char *var_name, char *buf, ...@@ -5337,7 +5380,8 @@ int mg_http_parse_header(struct mg_str *hdr, const char *var_name, char *buf,
#ifndef MG_DISABLE_FILESYSTEM #ifndef MG_DISABLE_FILESYSTEM
static int is_file_hidden(const char *path, static int is_file_hidden(const char *path,
const struct mg_serve_http_opts *opts) { const struct mg_serve_http_opts *opts,
int exclude_specials) {
const char *p1 = opts->per_directory_auth_file; const char *p1 = opts->per_directory_auth_file;
const char *p2 = opts->hidden_file_pattern; const char *p2 = opts->hidden_file_pattern;
...@@ -5347,8 +5391,9 @@ static int is_file_hidden(const char *path, ...@@ -5347,8 +5391,9 @@ static int is_file_hidden(const char *path,
path = pdir + 1; path = pdir + 1;
} }
return !strcmp(path, ".") || !strcmp(path, "..") || return (exclude_specials && (!strcmp(path, ".") || !strcmp(path, ".."))) ||
(p1 != NULL && !strcmp(path, p1)) || (p1 != NULL &&
mg_match_prefix(p1, strlen(p1), path) == (int) strlen(p1)) ||
(p2 != NULL && mg_match_prefix(p2, strlen(p2), path) > 0); (p2 != NULL && mg_match_prefix(p2, strlen(p2), path) > 0);
} }
...@@ -5452,7 +5497,7 @@ static int mg_http_check_digest_auth(struct http_message *hm, ...@@ -5452,7 +5497,7 @@ static int mg_http_check_digest_auth(struct http_message *hm,
static int is_authorized(struct http_message *hm, const char *path, static int is_authorized(struct http_message *hm, const char *path,
int is_directory, const char *domain, int is_directory, const char *domain,
const char *passwords_file, int is_global_pass_file) { const char *passwords_file, int is_global_pass_file) {
char buf[MAX_PATH_SIZE]; char buf[MG_MAX_PATH];
const char *p; const char *p;
FILE *fp; FILE *fp;
int authorized = 1; int authorized = 1;
...@@ -5464,11 +5509,9 @@ static int is_authorized(struct http_message *hm, const char *path, ...@@ -5464,11 +5509,9 @@ static int is_authorized(struct http_message *hm, const char *path,
snprintf(buf, sizeof(buf), "%s%c%s", path, DIRSEP, passwords_file); snprintf(buf, sizeof(buf), "%s%c%s", path, DIRSEP, passwords_file);
fp = fopen(buf, "r"); fp = fopen(buf, "r");
} else { } else {
if ((p = strrchr(path, '/')) == NULL && p = strrchr(path, DIRSEP);
(p = strrchr(path, '\\')) == NULL) { if (p == NULL) p = path;
p = path; snprintf(buf, sizeof(buf), "%.*s%c%s", (int) (p - path), path, DIRSEP,
}
snprintf(buf, sizeof(buf), "%.*s/%s", (int) (p - path), path,
passwords_file); passwords_file);
fp = fopen(buf, "r"); fp = fopen(buf, "r");
} }
...@@ -5479,6 +5522,8 @@ static int is_authorized(struct http_message *hm, const char *path, ...@@ -5479,6 +5522,8 @@ static int is_authorized(struct http_message *hm, const char *path,
} }
} }
DBG(("%s %s %d %d", path, passwords_file ? passwords_file : "",
is_global_pass_file, authorized));
return authorized; return authorized;
} }
#else #else
...@@ -5574,10 +5619,11 @@ static void scan_directory(struct mg_connection *nc, const char *dir, ...@@ -5574,10 +5619,11 @@ static void scan_directory(struct mg_connection *nc, const char *dir,
struct dirent *dp; struct dirent *dp;
DIR *dirp; DIR *dirp;
DBG(("%p [%s]", nc, dir));
if ((dirp = (opendir(dir))) != NULL) { if ((dirp = (opendir(dir))) != NULL) {
while ((dp = readdir(dirp)) != NULL) { while ((dp = readdir(dirp)) != NULL) {
/* Do not show current dir and hidden files */ /* Do not show current dir and hidden files */
if (is_file_hidden((const char *) dp->d_name, opts)) { if (is_file_hidden((const char *) dp->d_name, opts, 1)) {
continue; continue;
} }
snprintf(path, sizeof(path), "%s/%s", dir, dp->d_name); snprintf(path, sizeof(path), "%s/%s", dir, dp->d_name);
...@@ -5586,6 +5632,8 @@ static void scan_directory(struct mg_connection *nc, const char *dir, ...@@ -5586,6 +5632,8 @@ static void scan_directory(struct mg_connection *nc, const char *dir,
} }
} }
closedir(dirp); closedir(dirp);
} else {
DBG(("%p opendir(%s) -> %d", nc, dir, errno));
} }
} }
...@@ -5748,7 +5796,7 @@ static int remove_directory(const struct mg_serve_http_opts *opts, ...@@ -5748,7 +5796,7 @@ static int remove_directory(const struct mg_serve_http_opts *opts,
if ((dirp = opendir(dir)) == NULL) return 0; if ((dirp = opendir(dir)) == NULL) return 0;
while ((dp = readdir(dirp)) != NULL) { while ((dp = readdir(dirp)) != NULL) {
if (is_file_hidden((const char *) dp->d_name, opts)) { if (is_file_hidden((const char *) dp->d_name, opts, 1)) {
continue; continue;
} }
snprintf(path, sizeof(path), "%s%c%s", dir, '/', dp->d_name); snprintf(path, sizeof(path), "%s%c%s", dir, '/', dp->d_name);
...@@ -5890,43 +5938,35 @@ static int is_dav_request(const struct mg_str *s) { ...@@ -5890,43 +5938,35 @@ static int is_dav_request(const struct mg_str *s) {
* appended to the `path`, stat-ed, and result of `stat()` passed to `stp`. * appended to the `path`, stat-ed, and result of `stat()` passed to `stp`.
* If index file is not found, then `path` and `stp` remain unchanged. * If index file is not found, then `path` and `stp` remain unchanged.
*/ */
MG_INTERNAL int find_index_file(char *path, size_t path_len, const char *list, MG_INTERNAL void find_index_file(const char *path, const char *list,
cs_stat_t *stp) { char **index_file, cs_stat_t *stp) {
cs_stat_t st;
size_t n = strlen(path);
struct mg_str vec; struct mg_str vec;
size_t path_len = strlen(path);
int found = 0; int found = 0;
*index_file = NULL;
/* The 'path' given to us points to the directory. Remove all trailing */
/* directory separator characters from the end of the path, and */
/* then append single directory separator character. */
while (n > 0 && (path[n - 1] == '/' || path[n - 1] == '\\')) {
n--;
}
/* Traverse index files list. For each entry, append it to the given */ /* Traverse index files list. For each entry, append it to the given */
/* path and see if the file exists. If it exists, break the loop */ /* path and see if the file exists. If it exists, break the loop */
while ((list = mg_next_comma_list_entry(list, &vec, NULL)) != NULL) { while ((list = mg_next_comma_list_entry(list, &vec, NULL)) != NULL) {
/* Prepare full path to the index file */ cs_stat_t st;
snprintf(path + n, path_len - n, "/%.*s", (int) vec.len, vec.p); size_t len = path_len + 1 + vec.len + 1;
path[path_len - 1] = '\0'; *index_file = (char *) MG_REALLOC(*index_file, len);
if (*index_file == NULL) break;
snprintf(*index_file, len, "%s%c%.*s", path, DIRSEP, (int) vec.len, vec.p);
/* Does it exist? */ /* Does it exist? Is it a file? */
if (!mg_stat(path, &st)) { if (mg_stat(*index_file, &st) == 0 && S_ISREG(st.st_mode)) {
/* Yes it does, break the loop */ /* Yes it does, break the loop */
*stp = st; *stp = st;
found = 1; found = 1;
break; break;
} }
} }
/* If no index file exists, restore directory path, keep trailing slash. */
if (!found) { if (!found) {
path[n] = '\0'; MG_FREE(*index_file);
strncat(path + n, "/", path_len - n); *index_file = NULL;
} }
DBG(("[%s] [%s]", path, (*index_file ? *index_file : "")));
return found;
} }
static int send_port_based_redirect(struct mg_connection *c, static int send_port_based_redirect(struct mg_connection *c,
...@@ -5957,39 +5997,155 @@ static int send_port_based_redirect(struct mg_connection *c, ...@@ -5957,39 +5997,155 @@ static int send_port_based_redirect(struct mg_connection *c,
return 0; return 0;
} }
static void uri_to_path(struct http_message *hm, char *buf, size_t buf_len, MG_INTERNAL int mg_uri_to_local_path(struct http_message *hm,
const struct mg_serve_http_opts *opts) { const struct mg_serve_http_opts *opts,
char uri[MG_MAX_PATH]; char **local_path,
struct mg_str a, b, *host_hdr = mg_get_http_header(hm, "Host"); struct mg_str *remainder) {
int ok = 1;
const char *cp = hm->uri.p, *cp_end = hm->uri.p + hm->uri.len;
struct mg_str root = {NULL, 0};
const char *file_uri_start = cp;
*local_path = NULL;
remainder->p = NULL;
remainder->len = 0;
{ /* 1. Determine which root to use. */
const char *rewrites = opts->url_rewrites; const char *rewrites = opts->url_rewrites;
struct mg_str *hh = mg_get_http_header(hm, "Host");
mg_url_decode(hm->uri.p, hm->uri.len, uri, sizeof(uri), 0); struct mg_str a, b;
remove_double_dots(uri); /* Check rewrites first. */
snprintf(buf, buf_len, "%s%s", opts->document_root, uri); while ((rewrites = mg_next_comma_list_entry(rewrites, &a, &b)) != NULL) {
if (a.len > 1 && a.p[0] == '@') {
#ifndef MG_DISABLE_DAV /* Host rewrite. */
if (is_dav_request(&hm->method) && opts->dav_document_root != NULL) { if (hh != NULL && hh->len == a.len - 1 &&
snprintf(buf, buf_len, "%s%s", opts->dav_document_root, uri); mg_ncasecmp(a.p + 1, hh->p, a.len - 1) == 0) {
root = b;
break;
}
} else {
/* Regular rewrite, URI=directory */
int match_len = mg_match_prefix(a.p, a.len, cp);
if (match_len > 0) {
file_uri_start = cp + match_len;
if (*file_uri_start == '/' || file_uri_start == cp_end) {
/* Match ended at component boundary, ok. */
} else if (*(file_uri_start - 1) == '/') {
/* Pattern ends with '/', backtrack. */
file_uri_start--;
} else {
/* No match: must fall on the component boundary. */
continue;
}
root = b;
break;
}
} }
}
/* If no rewrite rules matched, use DAV or regular document root. */
if (root.p == NULL) {
#ifndef MG_DISABLE_DAV
if (opts->dav_document_root != NULL && is_dav_request(&hm->method)) {
root.p = opts->dav_document_root;
root.len = strlen(opts->dav_document_root);
} else
#endif #endif
{
root.p = opts->document_root;
root.len = strlen(opts->document_root);
}
}
assert(root.p != NULL && root.len > 0);
}
/* Handle URL rewrites */ { /* 2. Find where in the canonical URI path the local path ends. */
while ((rewrites = mg_next_comma_list_entry(rewrites, &a, &b)) != NULL) { const char *u = file_uri_start + 1;
if (a.len > 1 && a.p[0] == '@' && host_hdr != NULL && char *lp = (char *) MG_MALLOC(root.len + hm->uri.len + 1);
host_hdr->len == a.len - 1 && char *lp_end = lp + root.len + hm->uri.len + 1;
mg_ncasecmp(a.p + 1, host_hdr->p, a.len - 1) == 0) { char *p = lp, *ps;
/* This is a virtual host rewrite: @domain.name=document_root_dir */ int exists = 1;
snprintf(buf, buf_len, "%.*s%s", (int) b.len, b.p, uri); if (lp == NULL) {
ok = 0;
goto out;
}
memcpy(p, root.p, root.len);
p += root.len;
if (*(p - 1) == DIRSEP) p--;
*p = '\0';
ps = p;
/* Chop off URI path components one by one and build local path. */
while (u <= cp_end) {
const char *next = u;
struct mg_str component;
if (exists) {
cs_stat_t st;
exists = (mg_stat(lp, &st) == 0);
if (exists && S_ISREG(st.st_mode)) {
/* We found the terminal, the rest of the URI (if any) is path_info.
*/
if (*(u - 1) == '/') u--;
break; break;
}
}
if (u >= cp_end) break;
parse_uri_component((const char **) &next, cp_end, '/', &component);
if (component.len > 0) {
int len;
memmove(p + 1, component.p, component.len);
len = mg_url_decode(p + 1, component.len, p + 1, lp_end - p - 1, 0);
if (len <= 0) {
ok = 0;
break;
}
component.p = p + 1;
component.len = len;
if (mg_vcmp(&component, ".") == 0) {
/* Yum. */
} else if (mg_vcmp(&component, "..") == 0) {
while (p > ps && *p != DIRSEP) p--;
*p = '\0';
} else { } else {
/* This is a usual rewrite, URI=directory */ size_t i;
int match_len = mg_match_prefix(a.p, a.len, uri); #ifdef _WIN32
if (match_len > 0) { /* On Windows, make sure it's valid Unicode (no funny stuff). */
snprintf(buf, buf_len, "%.*s%s", (int) b.len, b.p, uri + match_len); wchar_t buf[MG_MAX_PATH * 2];
if (to_wchar(component.p, buf, MG_MAX_PATH) == 0) {
DBG(("[%.*s] smells funny", (int) component.len, component.p));
ok = 0;
break;
}
#endif
*p++ = DIRSEP;
/* No NULs and DIRSEPs in the component (percent-encoded). */
for (i = 0; i < component.len; i++, p++) {
if (*p == '\0' || *p == DIRSEP
#ifdef _WIN32
/* On Windows, "/" is also accepted, so check for that too. */
||
*p == '/'
#endif
) {
ok = 0;
break; break;
} }
} }
} }
}
u = next;
}
if (ok) {
*local_path = lp;
remainder->p = u;
remainder->len = cp_end - u;
} else {
MG_FREE(lp);
}
}
out:
DBG(("'%.*s' -> '%s' + '%.*s'", (int) hm->uri.len, hm->uri.p,
*local_path ? *local_path : "", (int) remainder->len, remainder->p));
return ok;
} }
#ifndef MG_DISABLE_CGI #ifndef MG_DISABLE_CGI
...@@ -6221,10 +6377,11 @@ static void addenv2(struct cgi_env_block *blk, const char *name) { ...@@ -6221,10 +6377,11 @@ static void addenv2(struct cgi_env_block *blk, const char *name) {
} }
static void prepare_cgi_environment(struct mg_connection *nc, const char *prog, static void prepare_cgi_environment(struct mg_connection *nc, const char *prog,
const struct mg_str *path_info,
const struct http_message *hm, const struct http_message *hm,
const struct mg_serve_http_opts *opts, const struct mg_serve_http_opts *opts,
struct cgi_env_block *blk) { struct cgi_env_block *blk) {
const char *s, *slash; const char *s;
struct mg_str *h; struct mg_str *h;
char *p; char *p;
size_t i; size_t i;
...@@ -6260,27 +6417,22 @@ static void prepare_cgi_environment(struct mg_connection *nc, const char *prog, ...@@ -6260,27 +6417,22 @@ static void prepare_cgi_environment(struct mg_connection *nc, const char *prog,
hm->query_string.len == 0 ? "" : "?", (int) hm->query_string.len, hm->query_string.len == 0 ? "" : "?", (int) hm->query_string.len,
hm->query_string.p); hm->query_string.p);
/* SCRIPT_NAME */ s = hm->uri.p + hm->uri.len - path_info->len - 1;
#if 0 if (*s == '/') {
if (nc->path_info != NULL) { const char *base_name = strrchr(prog, DIRSEP);
addenv(blk, "SCRIPT_NAME=%.*s", addenv(blk, "SCRIPT_NAME=%.*s/%s", (int) (s - hm->uri.p), hm->uri.p,
(int) (strlen(ri->uri) - strlen(nc->path_info)), ri->uri); (base_name != NULL ? base_name + 1 : prog));
addenv(blk, "PATH_INFO=%s", nc->path_info);
} else { } else {
#endif addenv(blk, "SCRIPT_NAME=%.*s", (int) (s - hm->uri.p + 1), hm->uri.p);
s = strrchr(prog, '/');
slash = hm->uri.p + hm->uri.len;
while (slash > hm->uri.p && *slash != '/') {
slash--;
} }
addenv(blk, "SCRIPT_NAME=%.*s%s", (int) (slash - hm->uri.p), hm->uri.p, addenv(blk, "SCRIPT_FILENAME=%s", prog);
s == NULL ? prog : s);
#if 0 if (path_info != NULL && path_info->len > 0) {
addenv(blk, "PATH_INFO=%.*s", (int) path_info->len, path_info->p);
/* Not really translated... */
addenv(blk, "PATH_TRANSLATED=%.*s", (int) path_info->len, path_info->p);
} }
#endif
addenv(blk, "SCRIPT_FILENAME=%s", prog);
addenv(blk, "PATH_TRANSLATED=%s", prog);
addenv(blk, "HTTPS=%s", nc->ssl != NULL ? "on" : "off"); addenv(blk, "HTTPS=%s", nc->ssl != NULL ? "on" : "off");
if ((h = mg_get_http_header((struct http_message *) hm, "Content-Type")) != if ((h = mg_get_http_header((struct http_message *) hm, "Content-Type")) !=
...@@ -6367,7 +6519,6 @@ static void cgi_ev_handler(struct mg_connection *cgi_nc, int ev, ...@@ -6367,7 +6519,6 @@ static void cgi_ev_handler(struct mg_connection *cgi_nc, int ev,
struct http_message hm; struct http_message hm;
struct mg_str *h; struct mg_str *h;
parse_http_headers(io->buf, io->buf + io->len, io->len, &hm); parse_http_headers(io->buf, io->buf + io->len, io->len, &hm);
/*printf("=== %d [%.*s]\n", k, k, io->buf);*/
if (mg_get_http_header(&hm, "Location") != NULL) { if (mg_get_http_header(&hm, "Location") != NULL) {
mg_printf(nc, "%s", "HTTP/1.1 302 Moved\r\n"); mg_printf(nc, "%s", "HTTP/1.1 302 Moved\r\n");
} else if ((h = mg_get_http_header(&hm, "Status")) != NULL) { } else if ((h = mg_get_http_header(&hm, "Status")) != NULL) {
...@@ -6391,6 +6542,7 @@ static void cgi_ev_handler(struct mg_connection *cgi_nc, int ev, ...@@ -6391,6 +6542,7 @@ static void cgi_ev_handler(struct mg_connection *cgi_nc, int ev,
} }
static void handle_cgi(struct mg_connection *nc, const char *prog, static void handle_cgi(struct mg_connection *nc, const char *prog,
const struct mg_str *path_info,
const struct http_message *hm, const struct http_message *hm,
const struct mg_serve_http_opts *opts) { const struct mg_serve_http_opts *opts) {
struct proto_data_http *dp; struct proto_data_http *dp;
...@@ -6399,13 +6551,14 @@ static void handle_cgi(struct mg_connection *nc, const char *prog, ...@@ -6399,13 +6551,14 @@ static void handle_cgi(struct mg_connection *nc, const char *prog,
const char *p; const char *p;
sock_t fds[2]; sock_t fds[2];
prepare_cgi_environment(nc, prog, hm, opts, &blk); DBG(("%p [%s]", nc, prog));
prepare_cgi_environment(nc, prog, path_info, hm, opts, &blk);
/* /*
* CGI must be executed in its own directory. 'dir' must point to the * CGI must be executed in its own directory. 'dir' must point to the
* directory containing executable program, 'p' must point to the * directory containing executable program, 'p' must point to the
* executable program name relative to 'dir'. * executable program name relative to 'dir'.
*/ */
if ((p = strrchr(prog, '/')) == NULL) { if ((p = strrchr(prog, DIRSEP)) == NULL) {
snprintf(dir, sizeof(dir), "%s", "."); snprintf(dir, sizeof(dir), "%s", ".");
} else { } else {
snprintf(dir, sizeof(dir), "%.*s", (int) (p - prog), prog); snprintf(dir, sizeof(dir), "%.*s", (int) (p - prog), prog);
...@@ -6526,35 +6679,61 @@ static int is_creation_request(const struct http_message *hm) { ...@@ -6526,35 +6679,61 @@ static int is_creation_request(const struct http_message *hm) {
return mg_vcmp(&hm->method, "MKCOL") == 0 || mg_vcmp(&hm->method, "PUT") == 0; return mg_vcmp(&hm->method, "MKCOL") == 0 || mg_vcmp(&hm->method, "PUT") == 0;
} }
void mg_send_http_file(struct mg_connection *nc, char *path, MG_INTERNAL void mg_send_http_file(struct mg_connection *nc, char *path,
size_t path_buf_len, struct http_message *hm, const struct mg_str *path_info,
struct http_message *hm,
struct mg_serve_http_opts *opts) { struct mg_serve_http_opts *opts) {
int stat_result, is_directory, is_dav = is_dav_request(&hm->method); int exists, is_directory, is_dav = is_dav_request(&hm->method);
uint32_t remote_ip = ntohl(*(uint32_t *) &nc->sa.sin.sin_addr); int is_cgi;
char *index_file = NULL;
cs_stat_t st; cs_stat_t st;
DBG(("serving [%s]", path)); exists = (mg_stat(path, &st) == 0);
stat_result = mg_stat(path, &st); is_directory = exists && S_ISDIR(st.st_mode);
is_directory = !stat_result && S_ISDIR(st.st_mode);
if (mg_check_ip_acl(opts->ip_acl, remote_ip) != 1) { if (is_directory) find_index_file(path, opts->index_files, &index_file, &st);
/* Not allowed to connect */
nc->flags |= MG_F_CLOSE_IMMEDIATELY; is_cgi =
} else if (is_dav && opts->dav_document_root == NULL) { (mg_match_prefix(opts->cgi_file_pattern, strlen(opts->cgi_file_pattern),
index_file ? index_file : path) > 0);
DBG(("%p %.*s [%s] exists=%d is_dir=%d is_dav=%d is_cgi=%d index=%s", nc,
(int) hm->method.len, hm->method.p, path, exists, is_directory, is_dav,
is_cgi, index_file ? index_file : ""));
if (is_directory && hm->uri.p[hm->uri.len - 1] != '/' && !is_dav) {
mg_printf(nc,
"HTTP/1.1 301 Moved\r\nLocation: %.*s/\r\n"
"Content-Length: 0\r\n\r\n",
(int) hm->uri.len, hm->uri.p);
MG_FREE(index_file);
return;
}
/* If we have path_info, the only way to handle it is CGI. */
if (path_info->len > 0 && !is_cgi) {
send_http_error(nc, 501, NULL);
return;
} else if (is_cgi) {
#if !defined(MG_DISABLE_CGI)
handle_cgi(nc, index_file ? index_file : path, path_info, hm, opts);
#else
send_http_error(nc, 501, NULL);
#endif /* MG_DISABLE_CGI */
MG_FREE(index_file);
return;
}
if (is_dav && opts->dav_document_root == NULL) {
send_http_error(nc, 501, NULL); send_http_error(nc, 501, NULL);
} else if (!is_authorized(hm, path, is_directory, opts->auth_domain, } else if (!is_authorized(hm, path, is_directory, opts->auth_domain,
opts->global_auth_file, 1) || opts->global_auth_file, 1) ||
!is_authorized(hm, path, is_directory, opts->auth_domain, !is_authorized(hm, path, is_directory, opts->auth_domain,
opts->per_directory_auth_file, 0)) { opts->per_directory_auth_file, 0)) {
mg_send_digest_auth_request(nc, opts->auth_domain); mg_send_digest_auth_request(nc, opts->auth_domain);
} else if ((stat_result != 0 || is_file_hidden(path, opts)) && } else if ((!exists || is_file_hidden(path, opts, 0 /* specials are ok */)) &&
!is_creation_request(hm)) { !is_creation_request(hm)) {
mg_printf(nc, "%s", "HTTP/1.1 404 Not Found\r\nContent-Length: 0\r\n\r\n"); send_http_error(nc, 404, NULL);
} else if (is_directory && path[strlen(path) - 1] != '/' && !is_dav) {
mg_printf(nc,
"HTTP/1.1 301 Moved\r\nLocation: %.*s/\r\n"
"Content-Length: 0\r\n\r\n",
(int) hm->uri.len, hm->uri.p);
#ifndef MG_DISABLE_DAV #ifndef MG_DISABLE_DAV
} else if (!mg_vcmp(&hm->method, "PROPFIND")) { } else if (!mg_vcmp(&hm->method, "PROPFIND")) {
handle_propfind(nc, path, &st, hm, opts); handle_propfind(nc, path, &st, hm, opts);
...@@ -6581,35 +6760,36 @@ void mg_send_http_file(struct mg_connection *nc, char *path, ...@@ -6581,35 +6760,36 @@ void mg_send_http_file(struct mg_connection *nc, char *path,
#endif #endif
} else if (!mg_vcmp(&hm->method, "OPTIONS")) { } else if (!mg_vcmp(&hm->method, "OPTIONS")) {
send_options(nc); send_options(nc);
} else if (S_ISDIR(st.st_mode) && } else if (is_directory && index_file == NULL) {
!find_index_file(path, path_buf_len, opts->index_files, &st)) {
if (strcmp(opts->enable_directory_listing, "yes") == 0) {
#ifndef MG_DISABLE_DIRECTORY_LISTING #ifndef MG_DISABLE_DIRECTORY_LISTING
if (strcmp(opts->enable_directory_listing, "yes") == 0) {
send_directory_listing(nc, path, hm, opts); send_directory_listing(nc, path, hm, opts);
#else
send_http_error(nc, 501, NULL);
#endif
} else { } else {
send_http_error(nc, 403, NULL); send_http_error(nc, 403, NULL);
} }
} else if (mg_match_prefix(opts->cgi_file_pattern,
strlen(opts->cgi_file_pattern), path) > 0) {
#if !defined(MG_DISABLE_CGI)
handle_cgi(nc, path, hm, opts);
#else #else
send_http_error(nc, 501, NULL); send_http_error(nc, 501, NULL);
#endif /* MG_DISABLE_CGI */ #endif
} else if (mg_is_not_modified(hm, &st)) { } else if (mg_is_not_modified(hm, &st)) {
send_http_error(nc, 304, "Not Modified"); send_http_error(nc, 304, "Not Modified");
} else { } else {
mg_send_http_file2(nc, path, &st, hm, opts); mg_send_http_file2(nc, index_file ? index_file : path, &st, hm, opts);
} }
MG_FREE(index_file);
} }
void mg_serve_http(struct mg_connection *nc, struct http_message *hm, void mg_serve_http(struct mg_connection *nc, struct http_message *hm,
struct mg_serve_http_opts opts) { struct mg_serve_http_opts opts) {
char path[MG_MAX_PATH]; char *path = NULL;
struct mg_str *hdr; struct mg_str *hdr, path_info;
uint32_t remote_ip = ntohl(*(uint32_t *) &nc->sa.sin.sin_addr);
if (mg_check_ip_acl(opts.ip_acl, remote_ip) != 1) {
/* Not allowed to connect */
send_http_error(nc, 403, NULL);
nc->flags |= MG_F_SEND_AND_CLOSE;
return;
}
if (send_port_based_redirect(nc, hm, &opts)) { if (send_port_based_redirect(nc, hm, &opts)) {
return; return;
...@@ -6633,9 +6813,19 @@ void mg_serve_http(struct mg_connection *nc, struct http_message *hm, ...@@ -6633,9 +6813,19 @@ void mg_serve_http(struct mg_connection *nc, struct http_message *hm,
if (opts.index_files == NULL) { if (opts.index_files == NULL) {
opts.index_files = "index.html,index.htm,index.shtml,index.cgi,index.php"; opts.index_files = "index.html,index.htm,index.shtml,index.cgi,index.php";
} }
/* Normalize path - resolve "." and ".." (in-place). */
if (!mg_normalize_uri_path(&hm->uri, &hm->uri)) {
send_http_error(nc, 400, NULL);
return;
}
if (mg_uri_to_local_path(hm, &opts, &path, &path_info) == 0) {
send_http_error(nc, 404, NULL);
return;
}
mg_send_http_file(nc, path, &path_info, hm, &opts);
uri_to_path(hm, path, sizeof(path), &opts); MG_FREE(path);
mg_send_http_file(nc, path, sizeof(path), hm, &opts); path = NULL;
/* Close connection for non-keep-alive requests */ /* Close connection for non-keep-alive requests */
if (mg_vcmp(&hm->proto, "HTTP/1.1") != 0 || if (mg_vcmp(&hm->proto, "HTTP/1.1") != 0 ||
...@@ -6764,7 +6954,7 @@ size_t mg_parse_multipart(const char *buf, size_t buf_len, char *var_name, ...@@ -6764,7 +6954,7 @@ size_t mg_parse_multipart(const char *buf, size_t buf_len, char *var_name,
} }
#endif /* MG_DISABLE_HTTP */ #endif /* MG_DISABLE_HTTP */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/util.c" #line 0 "./src/util.c"
#endif #endif
/* /*
...@@ -7130,7 +7320,7 @@ struct mg_str mg_mk_str(const char *s) { ...@@ -7130,7 +7320,7 @@ struct mg_str mg_mk_str(const char *s) {
struct mg_str ret = {s, strlen(s)}; struct mg_str ret = {s, strlen(s)};
return ret; return ret;
} }
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/json-rpc.c" #line 0 "./src/json-rpc.c"
#endif #endif
/* Copyright (c) 2014 Cesanta Software Limited */ /* Copyright (c) 2014 Cesanta Software Limited */
...@@ -7291,7 +7481,7 @@ int mg_rpc_parse_reply(const char *buf, int len, struct json_token *toks, ...@@ -7291,7 +7481,7 @@ int mg_rpc_parse_reply(const char *buf, int len, struct json_token *toks,
} }
#endif /* MG_DISABLE_JSON_RPC */ #endif /* MG_DISABLE_JSON_RPC */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/mqtt.c" #line 0 "./src/mqtt.c"
#endif #endif
/* /*
...@@ -7592,7 +7782,7 @@ void mg_mqtt_disconnect(struct mg_connection *nc) { ...@@ -7592,7 +7782,7 @@ void mg_mqtt_disconnect(struct mg_connection *nc) {
} }
#endif /* MG_DISABLE_MQTT */ #endif /* MG_DISABLE_MQTT */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/mqtt-broker.c" #line 0 "./src/mqtt-broker.c"
#endif #endif
/* /*
...@@ -7765,7 +7955,7 @@ struct mg_mqtt_session *mg_mqtt_next(struct mg_mqtt_broker *brk, ...@@ -7765,7 +7955,7 @@ struct mg_mqtt_session *mg_mqtt_next(struct mg_mqtt_broker *brk,
} }
#endif /* MG_ENABLE_MQTT_BROKER */ #endif /* MG_ENABLE_MQTT_BROKER */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/dns.c" #line 0 "./src/dns.c"
#endif #endif
/* /*
...@@ -8125,7 +8315,7 @@ void mg_set_protocol_dns(struct mg_connection *nc) { ...@@ -8125,7 +8315,7 @@ void mg_set_protocol_dns(struct mg_connection *nc) {
} }
#endif /* MG_DISABLE_DNS */ #endif /* MG_DISABLE_DNS */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/dns-server.c" #line 0 "./src/dns-server.c"
#endif #endif
/* /*
...@@ -8199,7 +8389,7 @@ int mg_dns_reply_record(struct mg_dns_reply *reply, ...@@ -8199,7 +8389,7 @@ int mg_dns_reply_record(struct mg_dns_reply *reply,
} }
#endif /* MG_ENABLE_DNS_SERVER */ #endif /* MG_ENABLE_DNS_SERVER */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/resolv.c" #line 0 "./src/resolv.c"
#endif #endif
/* /*
...@@ -8458,7 +8648,7 @@ int mg_resolve_async_opt(struct mg_mgr *mgr, const char *name, int query, ...@@ -8458,7 +8648,7 @@ int mg_resolve_async_opt(struct mg_mgr *mgr, const char *name, int query,
} }
#endif /* MG_DISABLE_RESOLVE */ #endif /* MG_DISABLE_RESOLVE */
#ifdef NS_MODULE_LINES #ifdef MG_MODULE_LINES
#line 0 "./src/coap.c" #line 0 "./src/coap.c"
#endif #endif
/* /*
......
...@@ -181,7 +181,10 @@ typedef struct stat cs_stat_t; ...@@ -181,7 +181,10 @@ typedef struct stat cs_stat_t;
typedef struct _stati64 cs_stat_t; typedef struct _stati64 cs_stat_t;
#endif #endif
#ifndef S_ISDIR #ifndef S_ISDIR
#define S_ISDIR(x) ((x) &_S_IFDIR) #define S_ISDIR(x) (((x) &_S_IFMT) == _S_IFDIR)
#endif
#ifndef S_ISREG
#define S_ISREG(x) (((x) &_S_IFMT) == _S_IFREG)
#endif #endif
#define DIRSEP '\\' #define DIRSEP '\\'
...@@ -1278,6 +1281,8 @@ int mg_parse_uri(struct mg_str uri, struct mg_str *scheme, ...@@ -1278,6 +1281,8 @@ int mg_parse_uri(struct mg_str uri, struct mg_str *scheme,
unsigned int *port, struct mg_str *path, struct mg_str *query, unsigned int *port, struct mg_str *path, struct mg_str *query,
struct mg_str *fragment); struct mg_str *fragment);
int mg_normalize_uri_path(const struct mg_str *in, struct mg_str *out);
#ifdef __cplusplus #ifdef __cplusplus
} }
#endif /* __cplusplus */ #endif /* __cplusplus */
......
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