int uwsgi_postbuffer_do_in_mem(struct wsgi_request *wsgi_req) { size_t remains = wsgi_req->post_cl; int ret; char *ptr = wsgi_req->post_buffering_buf; while (remains > 0) { if (uwsgi.harakiri_options.workers > 0) { inc_harakiri(wsgi_req, uwsgi.harakiri_options.workers); } ssize_t rlen = wsgi_req->socket->proto_read_body(wsgi_req, ptr, remains); if (rlen > 0) { remains -= rlen; ptr += rlen; continue; } if (rlen == 0) { uwsgi_read_error0(remains); return -1; } if (rlen < 0) { if (errno == EAGAIN || errno == EWOULDBLOCK || errno == EINPROGRESS) { goto wait; } uwsgi_read_error(remains); wsgi_req->read_errors++; return -1; } wait: ret = uwsgi_wait_read_req(wsgi_req); if (ret > 0) { rlen = wsgi_req->socket->proto_read_body(wsgi_req, ptr, remains); if (rlen > 0) { remains -= rlen; ptr += rlen; continue; } } if (ret < 0) { uwsgi_read_error(remains); wsgi_req->read_errors++; return -1; } uwsgi_read_timeout(remains); return -1; } return 0; }
int uwsgi_postbuffer_do_in_disk(struct wsgi_request *wsgi_req) { size_t post_remains = wsgi_req->post_cl; int ret; int upload_progress_fd = -1; char *upload_progress_filename = NULL; wsgi_req->post_file = tmpfile(); if (!wsgi_req->post_file) { uwsgi_error("uwsgi_postbuffer_do_in_disk()/tmpfile()"); return -1; } if (uwsgi.upload_progress) { // first check for X-Progress-ID size // separator + 'X-Progress-ID' + '=' + uuid upload_progress_filename = uwsgi_upload_progress_create(wsgi_req, &upload_progress_fd); if (!upload_progress_filename) { uwsgi_log("invalid X-Progress-ID value: must be a UUID\n"); } } // manage buffered data and upload progress while (post_remains > 0) { // during post buffering we need to constantly reset the harakiri if (uwsgi.shared->options[UWSGI_OPTION_HARAKIRI] > 0) { inc_harakiri(uwsgi.shared->options[UWSGI_OPTION_SOCKET_TIMEOUT]); } // we use the already available post buffering buffer to read chunks.... size_t remains = UMIN(post_remains, uwsgi.post_buffering); // first try to read data (there could be something already available ssize_t rlen = wsgi_req->socket->proto_read_body(wsgi_req, wsgi_req->post_buffering_buf, remains); if (rlen > 0) goto write; if (rlen == 0) { uwsgi_read_error0(remains); goto end; } if (rlen < 0) { if (errno == EAGAIN || errno == EWOULDBLOCK || errno == EINPROGRESS) { goto wait; } uwsgi_read_error(remains); goto end; } wait: ret = uwsgi_wait_read_req(wsgi_req); if (ret > 0) { rlen = wsgi_req->socket->proto_read_body(wsgi_req, wsgi_req->post_buffering_buf, remains); if (rlen > 0) goto write; if (rlen == 0) { uwsgi_read_error0(remains); } else { uwsgi_read_error(remains); } goto end; } if (ret < 0) { uwsgi_read_error(remains); goto end; } uwsgi_read_timeout(remains); goto end; write: if (fwrite(wsgi_req->post_buffering_buf, rlen, 1, wsgi_req->post_file) != 1) { uwsgi_error("uwsgi_postbuffer_do_in_disk()/fwrite()"); goto end; } post_remains -= rlen; if (upload_progress_filename) { // stop updating it on errors if (uwsgi_upload_progress_update(wsgi_req, upload_progress_fd, post_remains)) { uwsgi_upload_progress_destroy(upload_progress_filename, upload_progress_fd); upload_progress_filename = NULL; } } } rewind(wsgi_req->post_file); if (upload_progress_filename) { uwsgi_upload_progress_destroy(upload_progress_filename, upload_progress_fd); } return 0; end: if (upload_progress_filename) { uwsgi_upload_progress_destroy(upload_progress_filename, upload_progress_fd); } return -1; }
char *uwsgi_request_body_read(struct wsgi_request *wsgi_req, ssize_t hint, ssize_t *rlen) { int ret = -1; size_t remains = hint; // return empty if no post_cl or pos >= post_cl and no residual data if ((!wsgi_req->post_cl || wsgi_req->post_pos >= wsgi_req->post_cl ) && !wsgi_req->post_readline_pos) { return uwsgi.empty; } // return the whole input if (remains <= 0) { remains = wsgi_req->post_cl; } // some residual data ? if (wsgi_req->post_readline_pos > 0) { if (remains <= (wsgi_req->post_readline_watermark - wsgi_req->post_readline_pos)) { *rlen = remains; char *buf = wsgi_req->post_readline_buf + wsgi_req->post_readline_pos; wsgi_req->post_readline_pos += remains; return buf; } // the hint is higher than residual data, let's copy it to read() memory and go on size_t avail = wsgi_req->post_readline_watermark - wsgi_req->post_readline_pos; // check if we have enough memory... if (avail > wsgi_req->post_read_buf_size) { char *tmp_buf = realloc(wsgi_req->post_read_buf, avail); if (!tmp_buf) { uwsgi_error("uwsgi_request_body_read()/realloc()"); *rlen = -1; return NULL; } wsgi_req->post_read_buf = tmp_buf; wsgi_req->post_read_buf_size = avail; if (!wsgi_req->post_warning && wsgi_req->post_read_buf_size > (uwsgi.body_read_warning * 1024*1024)) { uwsgi_log("[uwsgi-warning] you are using read() on request body allocating over than %llu MB, that is really bad and can be avoided...\n", (unsigned long long) (wsgi_req->post_read_buf_size/(1024*1024))); wsgi_req->post_warning = 1; } } // fix remains... if (remains > 0) { remains -= avail; } *rlen += avail; memcpy(wsgi_req->post_read_buf, wsgi_req->post_readline_buf + wsgi_req->post_readline_pos, avail); wsgi_req->post_readline_pos = 0; wsgi_req->post_readline_watermark = 0; } if (remains + wsgi_req->post_pos > wsgi_req->post_cl) { remains = wsgi_req->post_cl - wsgi_req->post_pos; } if (remains == 0) { if (*rlen > 0) { return wsgi_req->post_read_buf; } else { return uwsgi.empty; } } // read from post buffering memory if (uwsgi.post_buffering > 0 && !wsgi_req->post_file) { *rlen += remains; char *buf = wsgi_req->post_buffering_buf+wsgi_req->post_pos; wsgi_req->post_pos += remains; return buf; } // ok we need to check if we need to allocate memory if (!wsgi_req->post_read_buf) { wsgi_req->post_read_buf = malloc(remains); if (!wsgi_req->post_read_buf) { uwsgi_error("uwsgi_request_body_read()/malloc()"); *rlen = -1; return NULL; } wsgi_req->post_read_buf_size = remains; } // need to realloc ? else { if ((remains+*rlen) > wsgi_req->post_read_buf_size) { char *tmp_buf = realloc(wsgi_req->post_read_buf, (remains+*rlen)); if (!tmp_buf) { uwsgi_error("uwsgi_request_body_read()/realloc()"); *rlen = -1; return NULL; } wsgi_req->post_read_buf = tmp_buf; wsgi_req->post_read_buf_size = (remains+*rlen); if (!wsgi_req->post_warning && wsgi_req->post_read_buf_size > (uwsgi.body_read_warning * 1024*1024)) { uwsgi_log("[uwsgi-warning] you are using read() on request body allocating over than %llu MB, that is really bad and can be avoided...\n", (unsigned long long) (wsgi_req->post_read_buf_size/(1024*1024))); wsgi_req->post_warning = 1; } } } // check for disk buffered body first (they are all read in one shot) if (wsgi_req->post_file) { if (fread(wsgi_req->post_read_buf + *rlen, remains, 1, wsgi_req->post_file) != 1) { *rlen = -1; uwsgi_error("uwsgi_request_body_read()/fread()"); return NULL; } *rlen += remains; wsgi_req->post_pos+= remains; return wsgi_req->post_read_buf; } // ok read all the required bytes... while(remains > 0) { // here we first try to read (as data could be already available) ssize_t len = wsgi_req->socket->proto_read_body(wsgi_req, wsgi_req->post_read_buf + *rlen , remains); if (len > 0) { wsgi_req->post_pos+=len; remains -= len; *rlen += len; continue; } // client closed connection... if (len == 0) { *rlen = -1; uwsgi_read_error0(remains); return NULL; } if (len < 0) { if (errno == EAGAIN || errno == EWOULDBLOCK || errno == EINPROGRESS) { goto wait; } *rlen = -1; uwsgi_read_error(remains); return NULL; } wait: ret = uwsgi_wait_read_req(wsgi_req); if (ret > 0) { len = wsgi_req->socket->proto_read_body(wsgi_req, wsgi_req->post_read_buf + *rlen, remains); if (len > 0) { wsgi_req->post_pos+=len; remains -= len; *rlen += len; continue; } *rlen = -1; if (len == 0) { uwsgi_read_error0(remains); } else { uwsgi_read_error(remains); } return NULL; } // 0 means timeout else if (ret == 0) { *rlen = 0; uwsgi_read_timeout(remains); return NULL; } *rlen = -1; uwsgi_read_error(remains); return NULL; } return wsgi_req->post_read_buf; }