mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-07 11:56:51 +00:00
3837 lines
107 KiB
C
3837 lines
107 KiB
C
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
*
|
|
* The contents of this file are subject to the Netscape Public License
|
|
* Version 1.0 (the "NPL"); you may not use this file except in
|
|
* compliance with the NPL. You may obtain a copy of the NPL at
|
|
* http://www.mozilla.org/NPL/
|
|
*
|
|
* Software distributed under the NPL is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
|
|
* for the specific language governing rights and limitations under the
|
|
* NPL.
|
|
*
|
|
* The Initial Developer of this code under the NPL is Netscape
|
|
* Communications Corporation. Portions created by Netscape are
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
|
|
* Reserved.
|
|
*/
|
|
/*
|
|
* state machine to speak HTTP
|
|
*
|
|
* Designed and implemented by Lou Montulli '94
|
|
* Additions/Changes by Judson Valeski, Gagan Saksena 1997
|
|
*/
|
|
#include "rosetta.h"
|
|
#include "xp.h"
|
|
#include "netutils.h"
|
|
#include "mkselect.h"
|
|
#include "mktcp.h"
|
|
#include "mkpadpac.h"
|
|
#include "prerror.h"
|
|
|
|
#include "merrors.h"
|
|
|
|
#include "mkgeturl.h"
|
|
#include "httpurl.h"
|
|
#include "shist.h"
|
|
#include "glhist.h"
|
|
#include "mkparse.h"
|
|
#include "mkstream.h"
|
|
#include "mkformat.h"
|
|
#include "mkaccess.h"
|
|
#include "cookies.h"
|
|
#include "netcache.h"
|
|
#include "mkautocf.h"
|
|
#include "secnav.h"
|
|
#include "ssl.h"
|
|
#include "jscookie.h"
|
|
#include "prefapi.h"
|
|
#include "xp_error.h"
|
|
#include "libi18n.h"
|
|
#include "prtime.h"
|
|
#include "prmem.h"
|
|
#include "plstr.h"
|
|
|
|
|
|
/* for XP_GetString() */
|
|
#include "xpgetstr.h"
|
|
extern int MK_REDIRECT_ATTEMPT_NOT_ALLOWED;
|
|
extern int MK_HTTP_TYPE_CONFLICT;
|
|
extern int MK_OUT_OF_MEMORY;
|
|
extern int MK_TCP_READ_ERROR;
|
|
extern int MK_TCP_WRITE_ERROR;
|
|
extern int MK_CONNECTION_REFUSED;
|
|
extern int MK_CONNECTION_TIMED_OUT;
|
|
extern int MK_UNABLE_TO_CONNECT;
|
|
extern int MK_UNABLE_TO_CONNECT_TO_PROXY;
|
|
extern int MK_UNABLE_TO_LOCATE_HOST;
|
|
extern int MK_UNABLE_TO_LOCATE_FILE;
|
|
extern int MK_UNABLE_TO_OPEN_FILE;
|
|
extern int MK_ZERO_LENGTH_FILE;
|
|
extern int MK_COMPUSERVE_AUTH_FAILED;
|
|
extern int XP_ALERT_UNKNOWN_STATUS;
|
|
extern int XP_ERRNO_EWOULDBLOCK;
|
|
extern int XP_PROGRESS_TRANSFER_DATA;
|
|
extern int XP_PROGRESS_TRYAGAIN;
|
|
extern int XP_PROGRESS_WAIT_REPLY;
|
|
extern int XP_HR_TRANSFER_INTERRUPTED;
|
|
extern int XP_TRANSFER_INTERRUPTED;
|
|
extern int XP_ERRNO_EIO;
|
|
|
|
|
|
#ifdef PROFILE
|
|
#pragma profile on
|
|
#endif
|
|
|
|
#define VERSION_STRING "HTTP/1.1"
|
|
|
|
#define MAX_CACHED_HTTP_CONNECTIONS 4
|
|
|
|
/* the maximum size of a HTTP servers first line of response
|
|
*/
|
|
#define MAX_FIRST_LINE_SIZE 250
|
|
|
|
/* temporary
|
|
*/
|
|
#if defined(XP_WIN) || defined(XP_OS2) /* IBM-SAH */
|
|
PUBLIC char *XP_AppName = 0;
|
|
PUBLIC char *XP_AppCodeName = 0;
|
|
PUBLIC char *XP_AppVersion = 0;
|
|
PUBLIC char *XP_AppLanguage = 0;
|
|
PUBLIC char *XP_AppPlatform = 0;
|
|
#else
|
|
PUBLIC CONST char *XP_AppName = 0;
|
|
PUBLIC CONST char *XP_AppCodeName = 0;
|
|
PUBLIC CONST char *XP_AppVersion = 0;
|
|
PUBLIC CONST char *XP_AppLanguage = 0;
|
|
PUBLIC CONST char *XP_AppPlatform = 0;
|
|
#endif
|
|
|
|
PUBLIC char * FE_UsersFromField=0; /* User's name/email address not used yet */
|
|
|
|
PRIVATE XP_List * http_connection_list=0;
|
|
PRIVATE IdentifyMeEnum http_identification_method = DoNotIdentifyMe;
|
|
PRIVATE Bool sendRefererHeader=TRUE;
|
|
|
|
/* definitions of state for the state machine design
|
|
*/
|
|
typedef enum {
|
|
HTTP_START_CONNECT,
|
|
HTTP_FINISH_CONNECT,
|
|
HTTP_SEND_PROXY_TUNNEL_REQUEST,
|
|
HTTP_BEGIN_UPLOAD_FILE,
|
|
HTTP_SEND_REQUEST,
|
|
HTTP_SEND_POST_DATA,
|
|
HTTP_PARSE_FIRST_LINE,
|
|
HTTP_PARSE_MIME_HEADERS,
|
|
HTTP_SETUP_STREAM,
|
|
HTTP_BEGIN_PUSH_PARTIAL_CACHE_FILE,
|
|
HTTP_PUSH_PARTIAL_CACHE_FILE,
|
|
HTTP_PULL_DATA,
|
|
HTTP_DONE,
|
|
HTTP_ERROR_DONE,
|
|
HG93634
|
|
HTTP_FREE
|
|
} StatesEnum;
|
|
|
|
/* structure to hold data about a tcp connection
|
|
* to a news host
|
|
*/
|
|
typedef struct _HTTPConnection {
|
|
char *hostname; /* hostname string (may contain :port) */
|
|
PRFileDesc *sock; /* socket */
|
|
XP_Bool busy; /* is the connection in use already? */
|
|
XP_Bool prev_cache; /* did this connection come from the cache? */
|
|
XP_Bool secure; /* is it a secure connection? */
|
|
} HTTPConnection;
|
|
|
|
typedef enum {
|
|
POINT_NINE,
|
|
ONE_POINT_O,
|
|
ONE_POINT_ONE
|
|
} HTTP_Version;
|
|
|
|
/* structure to hold data pertaining to the active state of
|
|
* a transfer in progress.
|
|
*
|
|
*/
|
|
typedef struct _HTTPConData {
|
|
StatesEnum next_state; /* the next state or action to be taken */
|
|
char * proxy_server; /* name of proxy server if any */
|
|
char * proxy_conf; /* proxy config ptr from proxy autoconfig */
|
|
char * line_buffer; /* temporary string storage */
|
|
char * server_headers; /* headers from the server for
|
|
* the proxy client
|
|
*/
|
|
char * orig_host; /* if the host gets modified
|
|
* for my "netscape" -> "www.netscape.com"
|
|
* hack, the original host gets put here
|
|
*/
|
|
XP_File partial_cache_fp;
|
|
int32 partial_needed; /* the part missing from the cache */
|
|
|
|
int32 total_size_of_files_to_post;
|
|
int32 total_amt_written;
|
|
|
|
int32 line_buffer_size; /* current size of the line buffer */
|
|
|
|
HTTPConnection *connection; /* struct to hold info about connection */
|
|
|
|
NET_StreamClass * stream; /* The output stream */
|
|
Bool pause_for_read; /* Pause now for next read? */
|
|
Bool send_http1; /* should we send http/1.1? */
|
|
Bool acting_as_proxy; /* are we acting as a proxy? */
|
|
Bool server_busy_retry; /* should we retry the get? */
|
|
Bool posting; /* are we posting? */
|
|
Bool doing_redirect; /* are we redirecting? */
|
|
Bool save_redirect_method; /* don't change METHOD when redirecting */
|
|
Bool sent_authorization; /* did we send auth with the request? */
|
|
Bool sent_proxy_auth; /* did we send proxy auth with the req? */
|
|
Bool authorization_required; /* did we get a 401 auth return code? */
|
|
Bool proxy_auth_required; /* did we get a 407 auth return code? */
|
|
Bool destroy_graph_progress; /* destroy graph progress? */
|
|
Bool destroy_file_upload_progress_dialog;
|
|
HTTP_Version protocol_version; /* .9 1.0 or 1.1 */
|
|
int32 original_content_length; /* the content length at the time of
|
|
* calling graph progress
|
|
*/
|
|
TCP_ConData *tcp_con_data; /* Data pointer for tcp connect state machine */
|
|
Bool use_proxy_tunnel; /* should we use a proxy tunnel? */
|
|
Bool proxy_tunnel_setup_done; /* is setup done */
|
|
Bool use_copy_from_cache; /* did we get a 304? */
|
|
Bool displayed_some_data; /* have we displayed any data? */
|
|
Bool save_connection; /* save this connection for reuse? */
|
|
Bool partial_cache_file;
|
|
Bool reuse_stream;
|
|
Bool connection_is_valid;
|
|
#ifdef XP_WIN
|
|
Bool calling_netlib_all_the_time; /* is SetCallNetlibAllTheTime set? */
|
|
#endif
|
|
void *write_post_data_data; /* a data object
|
|
* for the WritePostData function
|
|
*/
|
|
} HTTPConData;
|
|
|
|
/* macro's to simplify variable names */
|
|
#define CD_NEXT_STATE cd->next_state
|
|
#define CD_PROXY_SERVER cd->proxy_server
|
|
#define CD_PROXY_CONF cd->proxy_conf
|
|
#define CD_LINE_BUFFER cd->line_buffer
|
|
#define CD_SERVER_HEADERS cd->server_headers
|
|
#define CD_LINE_BUFFER_SIZE cd->line_buffer_size
|
|
#define CD_STREAM cd->stream
|
|
#define CD_SEND_HTTP1 cd->send_http1
|
|
#define CD_PAUSE_FOR_READ cd->pause_for_read
|
|
#define CD_ACTING_AS_PROXY cd->acting_as_proxy
|
|
#define CD_SERVER_BUSY_RETRY cd->server_busy_retry
|
|
#define CD_POSTING cd->posting
|
|
#define CD_DOING_REDIRECT cd->doing_redirect
|
|
#define CD_SENT_AUTHORIZATION cd->sent_authorization
|
|
#define CD_SENT_PROXY_AUTH cd->sent_proxy_auth
|
|
#define CD_AUTH_REQUIRED cd->authorization_required
|
|
#define CD_PROXY_AUTH_REQUIRED cd->proxy_auth_required
|
|
#define CD_DESTROY_GRAPH_PROGRESS cd->destroy_graph_progress
|
|
#define CD_ORIGINAL_CONTENT_LENGTH cd->original_content_length
|
|
#define CD_TCP_CON_DATA cd->tcp_con_data
|
|
#define CD_USE_PROXY_TUNNEL cd->use_proxy_tunnel
|
|
#define CD_PROXY_TUNNEL_SETUP_DONE cd->proxy_tunnel_setup_done
|
|
#define CD_USE_COPY_FROM_CACHE cd->use_copy_from_cache
|
|
#define CD_DISPLAYED_SOME_DATA cd->displayed_some_data
|
|
|
|
#define CE_URL_S ce->URL_s
|
|
#define CE_SOCK ce->socket
|
|
#define CE_CON_SOCK ce->con_sock
|
|
#define CE_STATUS ce->status
|
|
#define CE_WINDOW_ID ce->window_id
|
|
#define CE_BYTES_RECEIVED ce->bytes_received
|
|
#define CE_FORMAT_OUT ce->format_out
|
|
|
|
/* forward decl */
|
|
PRIVATE int32 net_ProcessHTTP (ActiveEntry *ce);
|
|
|
|
/*
|
|
* replace
|
|
* return(CE_STATUS)
|
|
* with
|
|
* RETURN_CE_STATUS
|
|
*/
|
|
PRIVATE
|
|
int ReturnErrorStatus (int status)
|
|
{
|
|
if (status < 0)
|
|
status |= status; /* set a breakpoint HERE to find errors */
|
|
return status;
|
|
}
|
|
#define STATUS(Status) ReturnErrorStatus (Status)
|
|
|
|
#define PUTBLOCK(b, l) (*cd->stream->put_block) \
|
|
(cd->stream, b, l)
|
|
#define PUTSTRING(s) (*cd->stream->put_block) \
|
|
(cd->stream, s, PL_strlen(s))
|
|
#define COMPLETE_STREAM (*cd->stream->complete) \
|
|
(cd->stream)
|
|
#define ABORT_STREAM(s) (*cd->stream->abort) \
|
|
(cd->stream, s)
|
|
PUBLIC void
|
|
NET_SetSendRefererHeader(Bool b)
|
|
{
|
|
sendRefererHeader=b;
|
|
}
|
|
|
|
/* set the method that the user wishes to identify themselves
|
|
* with.
|
|
* Default is DoNotIdentify unless this is called at startup
|
|
*/
|
|
PUBLIC void
|
|
NET_SetIdentifyMeType(IdentifyMeEnum method)
|
|
{
|
|
http_identification_method = method;
|
|
}
|
|
|
|
/* look for names like "ford" and "netscape"
|
|
* and turn them into "www.ford.com" and "www.netscape.com"
|
|
* If goBrowsing is enabled, this feature is turned off and the browser
|
|
* asks the search provider to resolve the word into a url. This is
|
|
* is prevent a kid typing in bambi (looking for the deer) from being
|
|
* taken to a porn site.
|
|
* returns 0 if found and replaced
|
|
*/
|
|
|
|
PRIVATE int
|
|
net_check_for_company_hostname(ActiveEntry *ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
Bool add_www = FALSE;
|
|
Bool add_com = FALSE;
|
|
Bool goBrowsing = FALSE;
|
|
|
|
if (PREF_GetBoolPref("browser.goBrowsingEnabled", &goBrowsing) != PREF_OK) goBrowsing = 0;
|
|
|
|
if(!cd->orig_host)
|
|
{
|
|
char *dot=NULL;
|
|
|
|
/* if the hostname doesn't have any
|
|
* dots in it, then assume it's of
|
|
* the form "netscape" or "ford".
|
|
* If we add www. and .com to the front
|
|
* and end we will get www.netscape.com :)
|
|
*/
|
|
char * host = NET_ParseURL(CE_URL_S->address, GET_HOST_PART);
|
|
|
|
if(host && *host && !(dot = PL_strchr(host, '.')))
|
|
{
|
|
add_www = add_com = TRUE;
|
|
}
|
|
else if(dot && !PL_strchr(dot+1, '.'))
|
|
{
|
|
/* there is only one dot in the host name
|
|
* so it's probably of the form of "netscape.com"
|
|
* or "ukans.edu". Add a "www." on the front
|
|
* and try again.
|
|
*/
|
|
add_www = TRUE;
|
|
}
|
|
|
|
if(add_www) {
|
|
/* no dots in hostname */
|
|
if (goBrowsing && !PL_strchr(CE_URL_S->address, '/')) {
|
|
char *pUrl;
|
|
PREF_CopyCharPref("network.search.url",&pUrl);
|
|
if (pUrl) {
|
|
char *tmp = NET_ParseURL(CE_URL_S->address, GET_HOST_PART);
|
|
char* new_address = PR_smprintf("%sgo+%s", pUrl, tmp);
|
|
PR_Free(pUrl);
|
|
PR_Free(CE_URL_S->address);
|
|
PR_FREEIF(tmp);
|
|
CE_URL_S->address = new_address;
|
|
|
|
if(cd->connection->sock != NULL)
|
|
NET_TotalNumberOfOpenConnections--;
|
|
return (0);
|
|
} else return (-1);
|
|
} else {
|
|
|
|
char *new_address = NET_ParseURL(CE_URL_S->address,
|
|
GET_PROTOCOL_PART);
|
|
char *tmp = NET_ParseURL(CE_URL_S->address, GET_HOST_PART);
|
|
|
|
/* save the host for error messages
|
|
*/
|
|
cd->orig_host = tmp;
|
|
|
|
StrAllocCat(new_address, "//www.");
|
|
StrAllocCat(new_address, tmp);
|
|
|
|
if(add_com)
|
|
StrAllocCat(new_address, ".com");
|
|
|
|
tmp = NET_ParseURL(CE_URL_S->address,
|
|
GET_PATH_PART | GET_SEARCH_PART | GET_HASH_PART);
|
|
|
|
StrAllocCat(new_address, tmp);
|
|
PR_FREEIF(tmp);
|
|
|
|
PR_Free(CE_URL_S->address);
|
|
CE_URL_S->address = new_address;
|
|
|
|
if(cd->connection->sock != NULL)
|
|
NET_TotalNumberOfOpenConnections--;
|
|
|
|
return(0); /* will repeat this step */
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* the host mapping trick has already been applied and failed
|
|
* redo the error message and fail
|
|
*/
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(
|
|
MK_UNABLE_TO_LOCATE_HOST,
|
|
cd->orig_host);
|
|
/* fall through for failure case */
|
|
}
|
|
|
|
return(-1);
|
|
}
|
|
|
|
/* begins the connect process
|
|
*
|
|
* returns the TCP status code
|
|
*/
|
|
PRIVATE int
|
|
net_start_http_connect(ActiveEntry * ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
Bool use_security=FALSE;
|
|
int def_port;
|
|
|
|
def_port = DEF_HTTP_PORT;
|
|
if(ce->protocol == SECURE_HTTP_TYPE_URL)
|
|
{
|
|
|
|
if(CD_PROXY_SERVER)
|
|
{
|
|
CD_USE_PROXY_TUNNEL = TRUE;
|
|
CD_PROXY_TUNNEL_SETUP_DONE = FALSE;
|
|
|
|
TRACEMSG(("HTTP: Using proxy tunnel"));
|
|
|
|
/* put in code to check for using a proxy tunnel
|
|
* if we need to use proxy tunnel set the options
|
|
* on the next two lines
|
|
*
|
|
* use_security = TRUE;
|
|
* def_port = DEF_HTTPS_PORT;
|
|
*
|
|
* @@@@@
|
|
*/
|
|
}
|
|
else
|
|
{
|
|
use_security = TRUE;
|
|
def_port = DEF_HTTPS_PORT;
|
|
}
|
|
}
|
|
|
|
/* if proxy_server is non NULL then use the string as a host:port
|
|
* when a proxy server is used a connection is made to the proxy
|
|
* host and port and the entire URL is sent instead of just
|
|
* the path part.
|
|
*/
|
|
if(CD_PROXY_SERVER)
|
|
{
|
|
/* MKConnect can take a URL or a host name as it's first
|
|
* argument
|
|
*/
|
|
CE_STATUS = NET_BeginConnect (CD_PROXY_SERVER,
|
|
NULL,
|
|
"HTTP",
|
|
def_port,
|
|
&cd->connection->sock,
|
|
use_security,
|
|
&CD_TCP_CON_DATA,
|
|
CE_WINDOW_ID,
|
|
&CE_URL_S->error_msg,
|
|
ce->socks_host,
|
|
ce->socks_port);
|
|
}
|
|
else
|
|
{
|
|
CE_STATUS = NET_BeginConnect (CE_URL_S->address,
|
|
CE_URL_S->IPAddressString,
|
|
"HTTP",
|
|
def_port,
|
|
&cd->connection->sock,
|
|
use_security,
|
|
&CD_TCP_CON_DATA,
|
|
CE_WINDOW_ID,
|
|
&CE_URL_S->error_msg,
|
|
ce->socks_host,
|
|
ce->socks_port);
|
|
}
|
|
|
|
/* set this so mkgeturl can select on it */
|
|
CE_SOCK = cd->connection->sock;
|
|
|
|
if(cd->connection->sock != NULL)
|
|
NET_TotalNumberOfOpenConnections++;
|
|
|
|
if (CE_STATUS < 0)
|
|
{
|
|
if(CE_STATUS == MK_UNABLE_TO_LOCATE_HOST
|
|
&& 0 == net_check_for_company_hostname(ce))
|
|
{
|
|
/* no need to set the state since this
|
|
* is the state we want again
|
|
*/
|
|
if(cd->connection->sock != NULL)
|
|
{
|
|
PR_Close(cd->connection->sock);
|
|
cd->connection->sock = NULL;
|
|
CE_SOCK = NULL;
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
TRACEMSG(("HTTP: Unable to connect to host for `%s' (errno = %d).",
|
|
CE_URL_S->address, SOCKET_ERRNO));
|
|
/*
|
|
* Proxy failover
|
|
*/
|
|
if (CD_PROXY_CONF && CD_PROXY_SERVER &&
|
|
(CE_STATUS == MK_UNABLE_TO_CONNECT ||
|
|
CE_STATUS == MK_CONNECTION_TIMED_OUT ||
|
|
CE_STATUS == MK_CONNECTION_REFUSED ||
|
|
CE_STATUS == MK_UNABLE_TO_LOCATE_HOST))
|
|
{
|
|
#ifdef MOZILLA_CLIENT
|
|
if (pacf_get_proxy_addr(CE_WINDOW_ID, CD_PROXY_CONF,
|
|
&ce->proxy_addr,
|
|
&ce->socks_host,
|
|
&ce->socks_port))
|
|
{
|
|
CD_PROXY_SERVER = ce->proxy_addr;
|
|
return net_start_http_connect(ce);
|
|
}
|
|
#endif /* MOZILLA_CLIENT */
|
|
}
|
|
|
|
if(CE_STATUS == MK_UNABLE_TO_CONNECT && CD_PROXY_SERVER)
|
|
{
|
|
StrAllocCopy (CE_URL_S->error_msg,
|
|
XP_GetString(MK_UNABLE_TO_CONNECT_TO_PROXY));
|
|
CE_STATUS = MK_UNABLE_TO_CONNECT_TO_PROXY;
|
|
}
|
|
|
|
CD_NEXT_STATE = HTTP_ERROR_DONE;
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
|
|
if(CE_STATUS == MK_CONNECTED)
|
|
{
|
|
if(CD_USE_PROXY_TUNNEL)
|
|
CD_NEXT_STATE = HTTP_SEND_PROXY_TUNNEL_REQUEST;
|
|
else if(ce->URL_s->files_to_post)
|
|
CD_NEXT_STATE = HTTP_BEGIN_UPLOAD_FILE;
|
|
else
|
|
{
|
|
HG56817
|
|
CD_NEXT_STATE = HTTP_SEND_REQUEST;
|
|
}
|
|
TRACEMSG(("Connected to HTTP server on sock #%d",cd->connection->sock));
|
|
NET_SetReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
}
|
|
else
|
|
{
|
|
CD_NEXT_STATE = HTTP_FINISH_CONNECT;
|
|
CD_PAUSE_FOR_READ = TRUE;
|
|
CE_CON_SOCK = cd->connection->sock; /* set the socket for select */
|
|
NET_SetConnectSelect(CE_WINDOW_ID, CE_CON_SOCK);
|
|
}
|
|
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
|
|
/* begins the connect process
|
|
*
|
|
* returns the TCP status code
|
|
*/
|
|
PRIVATE int
|
|
net_finish_http_connect(ActiveEntry * ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
int def_port;
|
|
|
|
def_port = DEF_HTTP_PORT;
|
|
if(ce->protocol == SECURE_HTTP_TYPE_URL)
|
|
{
|
|
def_port = DEF_HTTPS_PORT;
|
|
}
|
|
|
|
/* if proxy_server is non NULL then use the string as a host:port
|
|
* when a proxy server is used a connection is made to the proxy
|
|
* host and port and the entire URL is sent instead of just
|
|
* the path part.
|
|
*/
|
|
if(CD_PROXY_SERVER)
|
|
{
|
|
/* MKConnect can take a URL or a host name as it's first
|
|
* argument
|
|
*/
|
|
CE_STATUS = NET_FinishConnect(CD_PROXY_SERVER,
|
|
"HTTP",
|
|
def_port,
|
|
&cd->connection->sock,
|
|
&CD_TCP_CON_DATA,
|
|
CE_WINDOW_ID,
|
|
&CE_URL_S->error_msg);
|
|
}
|
|
else
|
|
{
|
|
CE_STATUS = NET_FinishConnect(CE_URL_S->address,
|
|
"HTTP",
|
|
def_port,
|
|
&cd->connection->sock,
|
|
&CD_TCP_CON_DATA,
|
|
CE_WINDOW_ID,
|
|
&CE_URL_S->error_msg);
|
|
}
|
|
|
|
|
|
if (CE_STATUS < 0)
|
|
{
|
|
if(CE_STATUS == MK_UNABLE_TO_LOCATE_HOST
|
|
&& 0 == net_check_for_company_hostname(ce))
|
|
{
|
|
if(cd->connection->sock != NULL)
|
|
{
|
|
NET_ClearConnectSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
PR_Close(cd->connection->sock);
|
|
cd->connection->sock = NULL;
|
|
CE_SOCK = NULL;
|
|
CE_CON_SOCK = NULL;
|
|
}
|
|
cd->next_state = HTTP_START_CONNECT;
|
|
return(0);
|
|
}
|
|
|
|
NET_ClearConnectSelect(CE_WINDOW_ID, CE_CON_SOCK);
|
|
TRACEMSG(("HTTP: Unable to connect to host for `%s' (errno = %d).",
|
|
CE_URL_S->address, SOCKET_ERRNO));
|
|
|
|
/*
|
|
* Proxy failover
|
|
*/
|
|
if (CD_PROXY_CONF && CD_PROXY_SERVER &&
|
|
(CE_STATUS == MK_UNABLE_TO_CONNECT ||
|
|
CE_STATUS == MK_CONNECTION_TIMED_OUT ||
|
|
CE_STATUS == MK_CONNECTION_REFUSED ||
|
|
CE_STATUS == MK_UNABLE_TO_LOCATE_HOST))
|
|
{
|
|
#ifdef MOZILLA_CLIENT
|
|
if (pacf_get_proxy_addr(CE_WINDOW_ID, CD_PROXY_CONF,
|
|
&ce->proxy_addr,
|
|
&ce->socks_host,
|
|
&ce->socks_port))
|
|
{
|
|
CD_PROXY_SERVER = ce->proxy_addr;
|
|
return net_start_http_connect(ce);
|
|
}
|
|
#endif /* MOZILLA_CLIENT */
|
|
}
|
|
|
|
/* proxy autodiscovery failover needs to be silent. If we were trying to
|
|
* connect to a pad server and failed, don't report errors, just go on
|
|
* about your business. */
|
|
if(NET_LoadingPac() && NET_UsingPadPac() ) {
|
|
/* Don't try to use the pad pac again. */
|
|
net_UsePadPac(FALSE);
|
|
CD_NEXT_STATE=HTTP_DONE;
|
|
CE_STATUS=0;
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
|
|
if(CE_STATUS == MK_UNABLE_TO_CONNECT && CD_PROXY_SERVER)
|
|
{
|
|
StrAllocCopy (CE_URL_S->error_msg,
|
|
XP_GetString(MK_UNABLE_TO_CONNECT_TO_PROXY));
|
|
CE_STATUS = MK_UNABLE_TO_CONNECT_TO_PROXY;
|
|
}
|
|
|
|
CD_NEXT_STATE = HTTP_ERROR_DONE;
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
|
|
if(CE_STATUS == MK_CONNECTED)
|
|
{
|
|
NET_ClearConnectSelect(CE_WINDOW_ID, CE_CON_SOCK);
|
|
CE_CON_SOCK = NULL; /* reset the socket so we don't select on it */
|
|
/* set this so mkgeturl can select on it */
|
|
CE_SOCK = cd->connection->sock;
|
|
if(CD_USE_PROXY_TUNNEL)
|
|
CD_NEXT_STATE = HTTP_SEND_PROXY_TUNNEL_REQUEST;
|
|
else if(ce->URL_s->files_to_post)
|
|
CD_NEXT_STATE = HTTP_BEGIN_UPLOAD_FILE;
|
|
else
|
|
{
|
|
HG43241
|
|
CD_NEXT_STATE = HTTP_SEND_REQUEST;
|
|
}
|
|
NET_SetReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
TRACEMSG(("Connected to HTTP server on sock #%d",cd->connection->sock));
|
|
|
|
}
|
|
else
|
|
{
|
|
/* unregister the old CE_SOCK from the select list
|
|
* and register the new value in the case that it changes
|
|
*/
|
|
if(ce->con_sock != cd->connection->sock)
|
|
{
|
|
NET_ClearConnectSelect(ce->window_id, ce->con_sock);
|
|
ce->con_sock = cd->connection->sock;
|
|
NET_SetConnectSelect(ce->window_id, ce->con_sock);
|
|
}
|
|
|
|
CD_PAUSE_FOR_READ = TRUE;
|
|
}
|
|
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
|
|
PRIVATE int
|
|
net_send_proxy_tunnel_request (ActiveEntry *ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
char *host = NET_ParseURL(CE_URL_S->address, GET_HOST_PART);
|
|
char *command=0;
|
|
char *auth = NULL;
|
|
|
|
StrAllocCopy(command, "CONNECT ");
|
|
StrAllocCat(command, host);
|
|
|
|
if(!PL_strchr(host, ':'))
|
|
{
|
|
char small_buf[20];
|
|
sprintf(small_buf, ":%d", DEF_HTTPS_PORT);
|
|
StrAllocCat(command, small_buf);
|
|
}
|
|
|
|
StrAllocCat(command, " "VERSION_STRING"\n");
|
|
|
|
/*
|
|
* Check if proxy is requiring authorization.
|
|
* If NULL, not necessary, or the proxy will return 407 to
|
|
* require authorization.
|
|
*
|
|
*/
|
|
if (NULL != (auth=NET_BuildProxyAuthString(CE_WINDOW_ID,
|
|
CE_URL_S,
|
|
CD_PROXY_SERVER)))
|
|
{
|
|
char *line = (char *)PR_Malloc(strlen(auth) + 30);
|
|
if (line) {
|
|
sprintf(line, "Proxy-authorization: %s%c%c", auth, CR, LF);
|
|
StrAllocCat(command, line);
|
|
PR_Free(line);
|
|
}
|
|
CD_SENT_PROXY_AUTH = TRUE;
|
|
TRACEMSG(("HTTP: Sending proxy-authorization: %s", auth));
|
|
}
|
|
else
|
|
{
|
|
TRACEMSG(("HTTP: Not sending proxy authorization (yet)"));
|
|
}
|
|
|
|
{
|
|
char line[200];
|
|
sprintf(line, "User-Agent: %.100s/%.90s" CRLF CRLF,
|
|
XP_AppCodeName, XP_AppVersion);
|
|
StrAllocCat(command, line);
|
|
}
|
|
|
|
TRACEMSG(("Tx: %s", command));
|
|
|
|
SSL_SetSockPeerID(cd->connection->sock, command);
|
|
CE_STATUS = NET_HTTPNetWrite(cd->connection->sock, command, PL_strlen(command));
|
|
|
|
PR_Free(command);
|
|
|
|
CD_PAUSE_FOR_READ = TRUE;
|
|
|
|
CD_NEXT_STATE = HTTP_PARSE_FIRST_LINE;
|
|
|
|
return(CE_STATUS);
|
|
}
|
|
|
|
#define POST_DATA_BUFFER_SIZE 2048
|
|
|
|
PRIVATE int32
|
|
net_get_size_with_crlf( char *filename, XP_FileType file_type, XP_Bool add_crlf )
|
|
{
|
|
XP_File xpfileptr;
|
|
uint32 return_value = 0;
|
|
int line_length;
|
|
char *line;
|
|
char *buffer;
|
|
|
|
if (!add_crlf)
|
|
{
|
|
/* Don't need to make any cr/lf adjustment, just stat the file and
|
|
return the size. */
|
|
|
|
XP_StatStruct stat_entry;
|
|
if(-1 == XP_Stat(filename, &stat_entry, file_type))
|
|
return -1;
|
|
else
|
|
return stat_entry.st_size;
|
|
}
|
|
|
|
xpfileptr = XP_FileOpen( filename, file_type, XP_FILE_READ );
|
|
if (!xpfileptr)
|
|
return -1;
|
|
|
|
buffer = (char *) PR_Malloc(POST_DATA_BUFFER_SIZE);
|
|
if (!buffer)
|
|
return -1;
|
|
|
|
do {
|
|
line = XP_FileReadLine(buffer, POST_DATA_BUFFER_SIZE-5, xpfileptr);
|
|
if (!line)
|
|
break;
|
|
|
|
line_length = PL_strlen(line);
|
|
|
|
if (line_length > 1 && line[line_length-2] == CR && line[line_length-1] == LF)
|
|
{
|
|
/* already ok */
|
|
}
|
|
else if(line_length > 0 && (line[line_length-1] == LF || line[line_length-1] == CR))
|
|
{
|
|
/* increment to account for missing CR or missing LF */
|
|
line_length++;
|
|
}
|
|
|
|
return_value += line_length;
|
|
} while (line);
|
|
|
|
PR_Free( buffer );
|
|
|
|
return return_value;
|
|
}
|
|
|
|
/* begin sending a file
|
|
*/
|
|
PRIVATE int
|
|
net_begin_upload_file (ActiveEntry *ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *) ce->con_data;
|
|
char * filename;
|
|
char * file_to_post;
|
|
char * header=0;
|
|
char * last_slash;
|
|
char * status_msg;
|
|
int i;
|
|
int32 adjusted_file_size = -1;
|
|
|
|
if(ce->URL_s->server_status == 401)
|
|
{
|
|
/* retrying with auth, don't change the filename */
|
|
cd->next_state = HTTP_SEND_REQUEST;
|
|
return(0);
|
|
}
|
|
|
|
PR_ASSERT(ce->URL_s->files_to_post && ce->URL_s->files_to_post[0]);
|
|
if(!ce->URL_s->files_to_post || !ce->URL_s->files_to_post[0])
|
|
return MK_UNABLE_TO_LOCATE_FILE;
|
|
|
|
/* get a filename from the files_to_post array
|
|
*/
|
|
for(i=0; ce->URL_s->files_to_post[i]; i++)
|
|
; /* find the end */
|
|
|
|
PR_ASSERT(i>0);
|
|
|
|
file_to_post = ce->URL_s->files_to_post[i-1];
|
|
|
|
/* zero the file now so that we don't upload it again. */
|
|
ce->URL_s->files_to_post[i-1] = NULL;
|
|
|
|
#ifdef XP_MAC
|
|
filename = PL_strrchr(file_to_post, '/');
|
|
#elif defined(XP_WIN)
|
|
filename = PL_strrchr(file_to_post, '\\');
|
|
#else
|
|
filename = PL_strrchr(file_to_post, '/');
|
|
#endif
|
|
|
|
if(!filename)
|
|
filename = file_to_post;
|
|
else
|
|
filename++; /* go past delimiter */
|
|
|
|
/* get the size of the file adjusting for crlf conversion */
|
|
adjusted_file_size = net_get_size_with_crlf(file_to_post, xpFileToPost, ce->URL_s->add_crlf ? ce->URL_s->add_crlf[i-1] : FALSE);
|
|
if (-1 == adjusted_file_size)
|
|
{
|
|
ce->URL_s->error_msg = NET_ExplainErrorDetails(MK_UNABLE_TO_LOCATE_FILE, file_to_post);
|
|
PR_Free(file_to_post);
|
|
return MK_UNABLE_TO_LOCATE_FILE;
|
|
}
|
|
|
|
status_msg = PR_smprintf("Uploading file %s", filename);
|
|
if(status_msg)
|
|
{
|
|
NET_Progress(ce->window_id, status_msg);
|
|
PR_Free(status_msg);
|
|
}
|
|
|
|
/*#ifdef EDITOR
|
|
FE_SaveDialogSetFilename(ce->window_id, filename);
|
|
#endif */ /* EDITOR */
|
|
|
|
header = PR_smprintf("Content-Length: %ld" CRLF CRLF, adjusted_file_size);
|
|
|
|
|
|
PR_FREEIF(ce->URL_s->post_headers);
|
|
/* if header is null this won't crash */
|
|
ce->URL_s->post_headers = header;
|
|
|
|
/* If the destination URL is explicitly specified in the post_to array, use it, otherwise
|
|
* generate destination from URL_s->address and filename. hardts */
|
|
if (ce->URL_s->post_to && ce->URL_s->post_to[i - 1]) {
|
|
PR_Free(ce->URL_s->address); /* We don't use it at all. */
|
|
ce->URL_s->address = ce->URL_s->post_to[i - 1];
|
|
ce->URL_s->post_to[i - 1] = NULL; /* zero the element */
|
|
}
|
|
else {
|
|
/* strip the filename from the last slash and append the
|
|
* new filename to the URL
|
|
*/
|
|
last_slash = PL_strrchr(ce->URL_s->address, '/');
|
|
if(last_slash)
|
|
*last_slash = '\0';
|
|
StrAllocCat(ce->URL_s->address, "/");
|
|
StrAllocCat(ce->URL_s->address, filename);
|
|
}
|
|
|
|
#ifdef EDITOR
|
|
{
|
|
/* Strip out username and password. from address */
|
|
char *pLocation = NULL;
|
|
if (!NET_ParseUploadURL( ce->URL_s->address, &pLocation, NULL,NULL )) {
|
|
PR_ASSERT(0);
|
|
}
|
|
if (CLEAR_CACHE_BIT(ce->format_out) != FO_LOCATION_INDEPENDENCE)
|
|
FE_SaveDialogSetFilename(ce->window_id, pLocation);
|
|
PR_Free(pLocation);
|
|
}
|
|
#endif /* EDITOR */
|
|
|
|
/* make the method PUT */
|
|
ce->URL_s->method = URL_PUT_METHOD;
|
|
|
|
/* specify the file to be uploaded */
|
|
StrAllocCopy(ce->URL_s->post_data, file_to_post);
|
|
|
|
/* specify that the post data is a file */
|
|
ce->URL_s->post_data_is_file = TRUE;
|
|
|
|
/* do the request */
|
|
cd->next_state = HTTP_SEND_REQUEST;
|
|
|
|
PR_Free(file_to_post);
|
|
|
|
return(0);
|
|
}
|
|
|
|
/* build the HTTP request.
|
|
*
|
|
* mallocs a big string that must be free'd
|
|
* returns the string
|
|
*/
|
|
PRIVATE unsigned int
|
|
net_build_http_request (URL_Struct * URL_s,
|
|
FO_Present_Types format_out,
|
|
Bool http1,
|
|
char * proxy_server,
|
|
Bool posting,
|
|
MWContext * window_id,
|
|
char ** command,
|
|
Bool * sent_authorization,
|
|
Bool * sent_proxy_auth)
|
|
{
|
|
char line_buffer[4096]; /* buffer */
|
|
char *auth; /* authorization header string */
|
|
size_t csize; /* size of command */
|
|
const char *tempURL=NULL;
|
|
char *proxyServer=NULL;
|
|
|
|
/* begin the request
|
|
*/
|
|
switch(URL_s->method)
|
|
{
|
|
case URL_MKDIR_METHOD:
|
|
#define MKDIR_WORD "MKDIR "
|
|
BlockAllocCopy(*command, MKDIR_WORD, 5);
|
|
csize = sizeof(MKDIR_WORD)-1;
|
|
break;
|
|
|
|
case URL_DELETE_METHOD:
|
|
#define DELETE_WORD "DELETE "
|
|
#define RMDIR_WORD "RMDIR "
|
|
if(URL_s->is_directory)
|
|
{
|
|
BlockAllocCopy(*command, RMDIR_WORD, 6);
|
|
csize = sizeof(RMDIR_WORD)-1;
|
|
}
|
|
else
|
|
{
|
|
BlockAllocCopy(*command, DELETE_WORD, 7);
|
|
csize = sizeof(DELETE_WORD)-1;
|
|
}
|
|
break;
|
|
|
|
case URL_POST_METHOD:
|
|
#define POST_WORD "POST "
|
|
BlockAllocCopy(*command, POST_WORD, 5);
|
|
csize = sizeof(POST_WORD)-1;
|
|
break;
|
|
|
|
case URL_PUT_METHOD:
|
|
#define PUT_WORD "PUT "
|
|
BlockAllocCopy(*command, PUT_WORD, 4);
|
|
csize = sizeof(PUT_WORD)-1;
|
|
break;
|
|
|
|
case URL_HEAD_METHOD:
|
|
#define HEAD_WORD "HEAD "
|
|
BlockAllocCopy(*command, HEAD_WORD, 5);
|
|
csize = sizeof(HEAD_WORD)-1;
|
|
break;
|
|
|
|
case URL_INDEX_METHOD:
|
|
#define INDEX_WORD "INDEX "
|
|
BlockAllocCopy(*command, INDEX_WORD, 6);
|
|
csize = sizeof(INDEX_WORD)-1;
|
|
break;
|
|
|
|
case URL_MOVE_METHOD:
|
|
#define MOVE_WORD "MOVE "
|
|
BlockAllocCopy(*command, MOVE_WORD, 5);
|
|
csize = sizeof(MOVE_WORD)-1;
|
|
break;
|
|
|
|
case URL_COPY_METHOD:
|
|
#define COPY_WORD "COPY "
|
|
BlockAllocCopy(*command, COPY_WORD, 5);
|
|
csize = sizeof(COPY_WORD)-1;
|
|
break;
|
|
|
|
default:
|
|
PR_ASSERT(0);
|
|
/* fall through to GET */
|
|
|
|
case URL_GET_METHOD:
|
|
#define GET_WORD "GET "
|
|
BlockAllocCopy(*command, GET_WORD, 4);
|
|
csize = sizeof(GET_WORD)-1;
|
|
break;
|
|
|
|
case URL_GETPROPERTIES_METHOD:
|
|
#define GETPROPERTIES_WORD "GETPROPERTIES "
|
|
BlockAllocCopy(*command, GETPROPERTIES_WORD, 14);
|
|
csize = sizeof(GETPROPERTIES_WORD)-1;
|
|
break;
|
|
}
|
|
|
|
|
|
/* if we are using a proxy gateway copy in the entire URL
|
|
* minus the hash mark
|
|
*/
|
|
if(proxy_server)
|
|
{
|
|
|
|
char *url_minus_hash = NET_ParseURL(URL_s->address,
|
|
GET_PROTOCOL_PART
|
|
| GET_USERNAME_PART
|
|
| GET_PASSWORD_PART
|
|
| GET_HOST_PART
|
|
| GET_PATH_PART
|
|
| GET_SEARCH_PART);
|
|
if(url_minus_hash)
|
|
{
|
|
BlockAllocCat(*command, (size_t) csize, url_minus_hash, PL_strlen(url_minus_hash));
|
|
csize += PL_strlen(url_minus_hash);
|
|
PR_Free(url_minus_hash);
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
/* else use just the path and search stuff
|
|
*/
|
|
char *path = NET_ParseURL(URL_s->address,
|
|
GET_PATH_PART | GET_SEARCH_PART);
|
|
BlockAllocCat(*command, csize, path, PL_strlen(path));
|
|
csize += PL_strlen(path);
|
|
|
|
PR_Free(path);
|
|
}
|
|
|
|
if(http1)
|
|
{
|
|
BlockAllocCat(*command, csize, " ", 1);
|
|
csize += 1;
|
|
BlockAllocCat(*command, csize,
|
|
VERSION_STRING,
|
|
PL_strlen(VERSION_STRING));
|
|
csize += PL_strlen(VERSION_STRING);
|
|
/* finish the line */
|
|
BlockAllocCat(*command, csize, CRLF, 2); /* CR LF, as in rfc 977 */
|
|
csize += 2;
|
|
|
|
if ((URL_s->etag) && (URL_s->force_reload != NET_SUPER_RELOAD))
|
|
{
|
|
/* add the If-None-Match header */
|
|
PL_strcpy(line_buffer, "If-None-Match: \"");
|
|
BlockAllocCat(*command, csize,
|
|
line_buffer, PL_strlen(line_buffer));
|
|
csize += PL_strlen(line_buffer);
|
|
BlockAllocCat(*command, csize,
|
|
URL_s->etag,
|
|
PL_strlen(URL_s->etag));
|
|
csize += PL_strlen(URL_s->etag);
|
|
/* Closing " */
|
|
BlockAllocCat(*command, csize,
|
|
"\"",
|
|
1);
|
|
csize += 1;
|
|
BlockAllocCat(*command, csize, CRLF, 2);
|
|
csize += 2;
|
|
}
|
|
|
|
if(URL_s->last_modified && URL_s->force_reload != NET_SUPER_RELOAD)
|
|
{
|
|
/* add if modified since
|
|
*
|
|
* right now this is being added even if the server sent pragma: no-cache
|
|
* I'm not sure if this is the right thing to do but it seems
|
|
* very efficient since we can still use the cache.
|
|
*/
|
|
|
|
/* add tmp character storage for strings in order for assert
|
|
* to work properly in debug compile.
|
|
* IBM compiler does not put "\" to make " work properly.
|
|
*/
|
|
#ifndef AIX
|
|
char *tmp_str = "http";
|
|
#endif
|
|
{
|
|
/* A conversion from time_t to NSPR's int64 is needed in order to
|
|
* use the NSPR time functions. URL_Struct should really be changed
|
|
* to use the NSPR time type instead of time_t.
|
|
*/
|
|
#ifndef NSPR20
|
|
uint64 timeInSec;
|
|
uint64 timeInUSec;
|
|
uint64 secToUSec;
|
|
PRTime expandedTime;
|
|
#else
|
|
PRTime timeInSec;
|
|
PRTime timeInUSec;
|
|
PRTime secToUSec;
|
|
PRExplodedTime expandedTime;
|
|
#endif /* NSPR20 */
|
|
|
|
LL_I2L(timeInSec, URL_s->last_modified);
|
|
LL_I2L(secToUSec, PR_USEC_PER_SEC);
|
|
LL_MUL(timeInUSec, timeInSec, secToUSec);
|
|
#ifndef NSPR20
|
|
#ifndef XP_MAC
|
|
timeInUSec = PR_ToGMT(timeInUSec);
|
|
#endif /* XP_MAC */
|
|
PR_ExplodeTime(&expandedTime, timeInUSec);
|
|
#else
|
|
PR_ExplodeTime(timeInUSec, PR_GMTParameters, &expandedTime);
|
|
#endif /* NSPR20 */
|
|
PR_FormatTimeUSEnglish(line_buffer, 400,
|
|
"If-Modified-Since: %a, %d %b %Y %H:%M:%S GMT",
|
|
&expandedTime);
|
|
}
|
|
|
|
/* must not be zero for http URL's
|
|
* or else I screwed up the cache logic
|
|
*/
|
|
#ifndef AIX
|
|
PR_ASSERT(PL_strncasecmp(URL_s->address, tmp_str, 4)
|
|
|| URL_s->real_content_length > 0);
|
|
#endif
|
|
|
|
if(URL_s->real_content_length)
|
|
sprintf(&line_buffer[PL_strlen(line_buffer)],
|
|
"; length=%ld" CRLF,
|
|
URL_s->real_content_length);
|
|
BlockAllocCat(*command, csize, line_buffer, PL_strlen(line_buffer));
|
|
csize += PL_strlen(line_buffer);
|
|
|
|
/* reset the expires since we will want to
|
|
* either get a new one from the server or
|
|
* set it to zero
|
|
*/
|
|
URL_s->expires = 0;
|
|
}
|
|
|
|
if(URL_s->http_headers) /* use headers that were passed in */
|
|
{
|
|
|
|
BlockAllocCat(*command, csize, URL_s->http_headers, PL_strlen(URL_s->http_headers));
|
|
csize += PL_strlen(URL_s->http_headers);
|
|
}
|
|
else
|
|
{
|
|
/* For MOVE method. This contains the destination uri name */
|
|
int meth=URL_s->method;
|
|
if(URL_s->destination
|
|
&& (*(URL_s->destination) != '\0')
|
|
&& ( (meth == URL_MOVE_METHOD)
|
|
|| (meth == URL_COPY_METHOD) )
|
|
){
|
|
int len=0;
|
|
if(meth == URL_MOVE_METHOD) {
|
|
sprintf(line_buffer, "New-uri: %s", URL_s->destination);
|
|
len=PL_strlen(line_buffer);
|
|
BlockAllocCat(*command, csize, line_buffer, len);
|
|
csize+=len;
|
|
}
|
|
else if(meth == URL_COPY_METHOD) {
|
|
;/* some http copy syntax */
|
|
}
|
|
|
|
BlockAllocCat(*command, csize, CRLF, PL_strlen(CRLF));
|
|
csize += PL_strlen(CRLF);
|
|
}
|
|
/* sendRefererHeader is set in NET_SetSendRefererHeaderPref in mkhttp.c.
|
|
This condition is set via a javascript pref and was implemented to
|
|
settle some privacy/security issue */
|
|
if(sendRefererHeader)
|
|
{
|
|
int url_type = NET_URL_Type(URL_s->referer);
|
|
if(URL_s->referer
|
|
&& url_type != MAILBOX_TYPE_URL
|
|
&& url_type != IMAP_TYPE_URL
|
|
&& url_type != FILE_TYPE_URL)
|
|
{
|
|
TRACEMSG(("Sending referer field"));
|
|
PL_strcpy(line_buffer, "Referer: ");
|
|
BlockAllocCat(*command, csize, line_buffer, PL_strlen(line_buffer));
|
|
csize += PL_strlen(line_buffer);
|
|
BlockAllocCat(*command, csize, URL_s->referer,
|
|
PL_strlen(URL_s->referer));
|
|
csize += PL_strlen(URL_s->referer);
|
|
BlockAllocCat(*command, csize, CRLF, 2);
|
|
csize += 2;
|
|
}
|
|
}
|
|
|
|
assert (XP_AppCodeName);
|
|
assert (XP_AppVersion);
|
|
|
|
if(proxy_server)
|
|
PL_strcpy(line_buffer, "Proxy-Connection: Keep-Alive" CRLF);
|
|
else
|
|
PL_strcpy(line_buffer, "Connection: Keep-Alive" CRLF);
|
|
|
|
sprintf (&line_buffer[PL_strlen(line_buffer)],
|
|
"User-Agent: %.100s/%.90s" CRLF,
|
|
XP_AppCodeName, XP_AppVersion);
|
|
BlockAllocCat(*command, csize, line_buffer, PL_strlen(line_buffer));
|
|
csize += PL_strlen(line_buffer);
|
|
|
|
if(URL_s->force_reload)
|
|
{
|
|
BlockAllocCat(*command, csize, "Pragma: no-cache" CRLF, 18);
|
|
csize += 18;
|
|
}
|
|
|
|
if(URL_s->range_header)
|
|
{
|
|
#ifndef AIX
|
|
char *tmp_str = CRLF;
|
|
PR_ASSERT(!PL_strstr(URL_s->range_header, tmp_str));
|
|
#endif
|
|
#define REQUEST_RANGE_HEADER "Range: "
|
|
BlockAllocCat(*command, csize,
|
|
REQUEST_RANGE_HEADER,
|
|
PL_strlen(REQUEST_RANGE_HEADER));
|
|
csize += PL_strlen(REQUEST_RANGE_HEADER);
|
|
BlockAllocCat(*command, csize,
|
|
URL_s->range_header,
|
|
PL_strlen(URL_s->range_header));
|
|
csize += PL_strlen(URL_s->range_header);
|
|
BlockAllocCat(*command, csize,
|
|
CRLF, PL_strlen(CRLF));
|
|
csize += PL_strlen(CRLF);
|
|
}
|
|
|
|
#define OLD_RANGE_SUPPORT
|
|
#ifdef OLD_RANGE_SUPPORT
|
|
/* this is here for backwards compatibility with
|
|
* the old range spec
|
|
* It should be removed before final beta 2
|
|
*/
|
|
if(URL_s->range_header)
|
|
{
|
|
#ifndef AIX
|
|
char *tmp_str = CRLF;
|
|
PR_ASSERT(!PL_strstr(URL_s->range_header, tmp_str));
|
|
#endif
|
|
#undef REQUEST_RANGE_HEADER
|
|
#define REQUEST_RANGE_HEADER "Request-Range: "
|
|
BlockAllocCat(*command, csize,
|
|
REQUEST_RANGE_HEADER,
|
|
PL_strlen(REQUEST_RANGE_HEADER));
|
|
csize += PL_strlen(REQUEST_RANGE_HEADER);
|
|
BlockAllocCat(*command, csize,
|
|
URL_s->range_header,
|
|
PL_strlen(URL_s->range_header));
|
|
csize += PL_strlen(URL_s->range_header);
|
|
BlockAllocCat(*command, csize,
|
|
CRLF, PL_strlen(CRLF));
|
|
csize += PL_strlen(CRLF);
|
|
}
|
|
#endif
|
|
|
|
if(1)
|
|
{
|
|
char * host = NET_ParseURL(URL_s->address, GET_HOST_PART);
|
|
|
|
if(host)
|
|
{
|
|
int len;
|
|
|
|
#define HOST_HEADER "Host: "
|
|
BlockAllocCat(*command,
|
|
csize,
|
|
HOST_HEADER,
|
|
sizeof(HOST_HEADER)-1);
|
|
csize += sizeof(HOST_HEADER)-1;
|
|
len = PL_strlen(host);
|
|
BlockAllocCat(*command, csize, host, len);
|
|
|
|
csize += len;
|
|
|
|
PR_Free(host);
|
|
|
|
BlockAllocCat(*command, csize, CRLF, 2);
|
|
csize += 2;
|
|
}
|
|
}
|
|
|
|
#ifdef SEND_FROM_FIELD
|
|
sprintf(line_buffer, "From: %.256s%c%c",
|
|
FE_UsersMailAddress() ? FE_UsersMailAddress() : "unregistered", CR,LF);
|
|
BlockAllocCat(*command, csize, line_buffer, PL_strlen(line_buffer));
|
|
csize += PL_strlen(line_buffer);
|
|
#endif /* SEND_FROM_FIELD */
|
|
|
|
|
|
if(CLEAR_CACHE_BIT(format_out) != FO_INTERNAL_IMAGE)
|
|
{
|
|
/* send Accept: *(slash)* as well as the others
|
|
*/
|
|
sprintf(line_buffer, "Accept: %s, %s, %s, %s, %s, */*" CRLF,
|
|
IMAGE_GIF, IMAGE_XBM, IMAGE_JPG, IMAGE_PJPG, IMAGE_PNG);
|
|
}
|
|
else
|
|
{
|
|
sprintf(line_buffer, "Accept: %s, %s, %s, %s, %s" CRLF,
|
|
IMAGE_GIF, IMAGE_XBM, IMAGE_JPG, IMAGE_PJPG, IMAGE_PNG);
|
|
}
|
|
|
|
BlockAllocCat(*command, csize, line_buffer, PL_strlen(line_buffer));
|
|
csize += PL_strlen(line_buffer);
|
|
|
|
/* add Accept-Encoding Header */
|
|
sprintf(line_buffer, "Accept-Encoding: %s" CRLF, ENCODING_GZIP2);
|
|
BlockAllocCat(*command, csize, line_buffer, PL_strlen(line_buffer));
|
|
csize += PL_strlen(line_buffer);
|
|
|
|
#ifdef MOZILLA_CLIENT
|
|
#define SEND_ACCEPT_LANGUAGE 1
|
|
#ifdef SEND_ACCEPT_LANGUAGE /* Added by ftang */
|
|
{
|
|
char *acceptlang = INTL_GetAcceptLanguage();
|
|
if((acceptlang != NULL) && ( *acceptlang != '\0') )
|
|
{
|
|
sprintf(line_buffer, "Accept-Language: %s" CRLF,
|
|
acceptlang );
|
|
BlockAllocCat(*command, csize, line_buffer, PL_strlen(line_buffer));
|
|
csize += PL_strlen(line_buffer);
|
|
}
|
|
}
|
|
#endif /* SEND_ACCEPT_LANGUAGE */
|
|
|
|
#define SEND_ACCEPT_CHARSET 1
|
|
#ifdef SEND_ACCEPT_CHARSET /* Added by bstell */
|
|
{
|
|
char *acceptCharset = INTL_GetAcceptCharset();
|
|
if((acceptCharset != NULL) && ( *acceptCharset != '\0') )
|
|
{
|
|
sprintf(line_buffer, "Accept-Charset: %s" CRLF,
|
|
acceptCharset );
|
|
BlockAllocCat(*command, csize, line_buffer, PL_strlen(line_buffer));
|
|
csize += PL_strlen(line_buffer);
|
|
}
|
|
}
|
|
#endif /* SEND_ACCEPT_CHARSET */
|
|
#endif /* MOZILLA_CLIENT */
|
|
|
|
/*
|
|
* Check if proxy is requiring authorization.
|
|
* If NULL, not necessary, or the proxy will return 407 to
|
|
* require authorization.
|
|
*
|
|
*/
|
|
/* This was hacked in because proxy auth can be required when the Auto-config url
|
|
* itself requires authorization. We used to ask for proxy auth only when the proxy was
|
|
* input directly into the prefs. Now we check to see if there's a pacurl (if there
|
|
* is, there can't be a proxy url simultaneously) use it, otherwise we're using a
|
|
* proxy from the prefs.
|
|
*/
|
|
|
|
/* Figure out which kind of proxy we're using: PAC or straight proxy.
|
|
* DON'T FREE tempURL!!!
|
|
*/
|
|
if ( (tempURL = net_GetPACUrl()) && (*tempURL) )
|
|
proxyServer = NET_ParseURL(tempURL, GET_HOST_PART | GET_PATH_PART | GET_USERNAME_PART | GET_PASSWORD_PART);
|
|
else
|
|
proxyServer = proxy_server;
|
|
|
|
if (proxyServer &&
|
|
NULL != (auth=NET_BuildProxyAuthString(window_id,
|
|
URL_s,
|
|
proxyServer)))
|
|
{
|
|
if (tempURL)
|
|
PR_Free(proxyServer);
|
|
*sent_proxy_auth = TRUE;
|
|
sprintf(line_buffer, "Proxy-authorization: %.3840s%c%c", auth, CR, LF);
|
|
BlockAllocCat(*command, csize, line_buffer, PL_strlen(line_buffer));
|
|
csize += PL_strlen(line_buffer);
|
|
TRACEMSG(("HTTP: Sending proxy-authorization: %s", auth));
|
|
}
|
|
else
|
|
{
|
|
TRACEMSG(("HTTP: Not sending proxy authorization (yet)"));
|
|
}
|
|
|
|
/* NET_BuildAuthString will return non-NULL if authorization is
|
|
* known to be required for this document.
|
|
*
|
|
* If NULL is returned authorization is not required or is
|
|
* not known to be required yet so we will look for a 401
|
|
* return code later on to see if authorization will be
|
|
* neccessary
|
|
*/
|
|
if (NULL!=(auth=NET_BuildAuthString(window_id, URL_s)))
|
|
{
|
|
*sent_authorization = TRUE;
|
|
BlockAllocCat(*command, csize, auth, PL_strlen(auth));
|
|
csize += PL_strlen(auth);
|
|
TRACEMSG(("HTTP: Sending authorization: %s", auth));
|
|
}
|
|
else
|
|
{
|
|
TRACEMSG(("HTTP: Not sending authorization (yet)"));
|
|
}
|
|
|
|
if (NULL!=(auth=NET_GetCookie(window_id, URL_s->address)))
|
|
{
|
|
int len;
|
|
PL_strcpy(line_buffer, "Cookie: ");
|
|
BlockAllocCat(*command, csize,
|
|
line_buffer, PL_strlen(line_buffer));
|
|
|
|
csize += PL_strlen(line_buffer);
|
|
len = PL_strlen(auth);
|
|
BlockAllocCat(*command, csize, auth, len);
|
|
csize += len;
|
|
BlockAllocCat(*command, csize, CRLF, PL_strlen(CRLF));
|
|
csize += PL_strlen(CRLF);
|
|
TRACEMSG(("HTTP: Sending Cookie: %s", auth));
|
|
PR_Free(auth);
|
|
}
|
|
else
|
|
{
|
|
TRACEMSG(("HTTP: Not sending cookie"));
|
|
}
|
|
|
|
} /* end of if passed in headers */
|
|
} /* end of if(http1) */
|
|
|
|
/* end of Get/Post request */
|
|
|
|
/* Blank line means "end" of headers
|
|
* If we are posting WritePostData will
|
|
* add the extra line feed for us
|
|
*/
|
|
if(!posting)
|
|
{
|
|
BlockAllocCat(*command, csize, CRLF, 2);
|
|
csize += 2;
|
|
}
|
|
|
|
return((unsigned int) csize);
|
|
}
|
|
|
|
HG12515
|
|
|
|
/* send_http_request makes a connection to the http server
|
|
* and sends a request (ethier http1 or http/.9) to the server
|
|
*
|
|
* returns the TCP status code
|
|
*/
|
|
#define SOCK_CHUNK_SIZE 1024
|
|
PRIVATE int
|
|
net_send_http_request (ActiveEntry *ce)
|
|
{
|
|
/* assign it so that we have a typed structure pointer */
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
char * command=0;
|
|
unsigned int command_size;
|
|
|
|
/* we make the assumption that we are always acting as a proxy
|
|
* server if we get http_headers passed in the URL structure.
|
|
*
|
|
* when acting as a proxy_server the clients headers are sent
|
|
* to the server instead of being generated here and the headers
|
|
* from the server are unparsed and sent on the the client.
|
|
*/
|
|
if(ce->format_out == FO_OUT_TO_PROXY_CLIENT ||
|
|
ce->format_out == FO_CACHE_AND_OUT_TO_PROXY_CLIENT)
|
|
CD_ACTING_AS_PROXY=YES;
|
|
|
|
TRACEMSG(("Entered \"send_http_request\""));
|
|
|
|
/* we make the assumption that we will always use the POST
|
|
* method if there is post data in the URL structure
|
|
* Is that a bad assumption?
|
|
*/
|
|
if(CE_URL_S->method == URL_POST_METHOD
|
|
|| CE_URL_S->method == URL_PUT_METHOD) {
|
|
CD_POSTING = YES;
|
|
#ifdef MOZILLA_CLIENT
|
|
SECNAV_Posting(cd->connection->sock);
|
|
#endif /* MOZILLA_CLIENT */
|
|
} else if (CE_URL_S->method == URL_HEAD_METHOD) {
|
|
#ifdef MOZILLA_CLIENT
|
|
SECNAV_HTTPHead(cd->connection->sock);
|
|
#endif /* MOZILLA_CLIENT */
|
|
}
|
|
|
|
/* zero out associated mime fields in URL structure
|
|
*
|
|
* all except content_length since that may be passed in
|
|
*/
|
|
CE_URL_S->protection_template = 0;
|
|
PR_FREEIF(CE_URL_S->redirecting_url);
|
|
CE_URL_S->redirecting_url = NULL;
|
|
PR_FREEIF(CE_URL_S->authenticate);
|
|
CE_URL_S->authenticate = NULL;
|
|
PR_FREEIF(CE_URL_S->proxy_authenticate);
|
|
CE_URL_S->proxy_authenticate = NULL;
|
|
|
|
/* Build the request command. (It must be free'd!!!)
|
|
*
|
|
* build_http_request will assemble a string containing
|
|
* the main request line, HTTP/1.0 MIME headers and
|
|
* the post data (if it exits)
|
|
*/
|
|
command_size = net_build_http_request(CE_URL_S,
|
|
CE_FORMAT_OUT,
|
|
CD_SEND_HTTP1,
|
|
(CD_USE_PROXY_TUNNEL ? NULL : CD_PROXY_SERVER),
|
|
CD_POSTING,
|
|
CE_WINDOW_ID,
|
|
&command,
|
|
&CD_SENT_AUTHORIZATION,
|
|
&CD_SENT_PROXY_AUTH);
|
|
|
|
TRACEMSG(("Sending HTTP Request:\n---------------------------------"));
|
|
|
|
CE_STATUS = (int) NET_BlockingWrite(cd->connection->sock, command, command_size);
|
|
|
|
#if defined(JAVA)
|
|
#if defined(DEBUG)
|
|
if(MKLib_trace_flag)
|
|
{
|
|
NET_NTrace("Tx: ", 4);
|
|
NET_NTrace(command, command_size);
|
|
}
|
|
#endif /* DEBUG */
|
|
#endif /* JAVA */
|
|
PR_Free (command); /* freeing the request */
|
|
|
|
if (CE_STATUS < 0)
|
|
{
|
|
int err = PR_GetError();
|
|
|
|
if(err == PR_WOULD_BLOCK_ERROR)
|
|
return(1); /* continue */
|
|
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(MK_TCP_WRITE_ERROR, err);
|
|
|
|
CD_NEXT_STATE= HTTP_ERROR_DONE;
|
|
|
|
return(MK_TCP_WRITE_ERROR);
|
|
}
|
|
|
|
|
|
/* make sure these are empty
|
|
*/
|
|
PR_FREEIF(CD_LINE_BUFFER); /* reset */
|
|
CD_LINE_BUFFER = NULL;
|
|
CD_LINE_BUFFER_SIZE=0; /* reset */
|
|
|
|
if(CD_POSTING
|
|
&& CE_URL_S->post_data)
|
|
{
|
|
NET_ClearReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
NET_SetConnectSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
#ifdef XP_WIN
|
|
cd->calling_netlib_all_the_time = TRUE;
|
|
NET_SetCallNetlibAllTheTime(CE_WINDOW_ID, "mkhttp");
|
|
#endif /* XP_WIN */
|
|
CE_CON_SOCK = cd->connection->sock;
|
|
CD_NEXT_STATE = HTTP_SEND_POST_DATA;
|
|
return(0);
|
|
}
|
|
|
|
CD_NEXT_STATE = HTTP_PARSE_FIRST_LINE;
|
|
|
|
/* Don't pause for read any more because we need to do
|
|
* at least a single read right away to detect if the
|
|
* connection is bad. Apparently windows will queue a
|
|
* write and not return a failure, but in fact the
|
|
* connection has already been closed by the server.
|
|
* Windows select will not even detect the exception
|
|
* so we end up deadlocked. By doing one read we
|
|
* can detect errors immediately
|
|
*
|
|
* CD_PAUSE_FOR_READ = TRUE;
|
|
*/
|
|
|
|
{
|
|
char * nonProxyHost = NET_ParseURL(CE_URL_S->address, GET_HOST_PART);
|
|
if (nonProxyHost) {
|
|
char* msg = PR_smprintf(XP_GetString(XP_PROGRESS_WAIT_REPLY),
|
|
nonProxyHost);
|
|
if (msg) {
|
|
NET_Progress(CE_WINDOW_ID, msg);
|
|
PR_Free(msg);
|
|
}
|
|
PR_Free(nonProxyHost);
|
|
}
|
|
}
|
|
|
|
return STATUS(CE_STATUS);
|
|
|
|
} /* end of net_send_http_request */
|
|
|
|
PRIVATE int
|
|
net_http_send_post_data (ActiveEntry *ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
/* Default to not add_crlf_to_line_endings for HTTP. */
|
|
XP_Bool add_crlf = FALSE;
|
|
|
|
|
|
/* make sure the line buffer is large enough
|
|
* for us to use as a buffer
|
|
*/
|
|
if(cd->line_buffer_size < 200)
|
|
{
|
|
PR_FREEIF(cd->line_buffer);
|
|
cd->line_buffer = (char*)PR_Malloc(256);
|
|
cd->line_buffer_size = 256;
|
|
}
|
|
|
|
/* If the add_crlf field is set in the URL struct, find the
|
|
entry corresponding to the current file being posted. */
|
|
if (CE_URL_S->files_to_post && CE_URL_S->add_crlf) {
|
|
int n = 0;
|
|
while (CE_URL_S->files_to_post[n]) {
|
|
n++;
|
|
}
|
|
/* n should now point after the last entry in files_to_post,
|
|
which is the current file being uploaded. */
|
|
add_crlf = CE_URL_S->add_crlf[n];
|
|
}
|
|
|
|
/* returns 0 on done and negative on error
|
|
* positive if it needs to continue.
|
|
*/
|
|
CE_STATUS = NET_WritePostData(CE_WINDOW_ID, CE_URL_S,
|
|
cd->connection->sock,
|
|
&cd->write_post_data_data,
|
|
add_crlf);
|
|
|
|
cd->pause_for_read = TRUE;
|
|
|
|
if(CE_STATUS == 0)
|
|
{
|
|
/* normal done
|
|
*/
|
|
TRACEMSG(("End of post data data"));
|
|
|
|
/* make sure these are empty
|
|
*/
|
|
PR_FREEIF(cd->line_buffer); /* reset */
|
|
cd->line_buffer = NULL;
|
|
cd->line_buffer_size=0; /* reset */
|
|
|
|
NET_ClearConnectSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
NET_SetReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
CE_CON_SOCK = NULL;
|
|
|
|
{
|
|
char * nonProxyHost = NET_ParseURL(CE_URL_S->address, GET_HOST_PART);
|
|
if (nonProxyHost) {
|
|
char* msg = PR_smprintf(XP_GetString(XP_PROGRESS_WAIT_REPLY),
|
|
nonProxyHost);
|
|
if (msg) {
|
|
NET_Progress(CE_WINDOW_ID, msg);
|
|
PR_Free(msg);
|
|
}
|
|
PR_Free(nonProxyHost);
|
|
}
|
|
}
|
|
|
|
cd->next_state = HTTP_PARSE_FIRST_LINE;
|
|
return(0);
|
|
}
|
|
else if(cd->total_size_of_files_to_post && ce->status > 0)
|
|
{
|
|
cd->total_amt_written += ce->status;
|
|
|
|
FE_GraphProgress(ce->window_id,
|
|
ce->URL_s,
|
|
cd->total_amt_written,
|
|
ce->status,
|
|
cd->total_size_of_files_to_post);
|
|
FE_SetProgressBarPercent(ce->window_id,
|
|
cd->total_amt_written*100/cd->total_size_of_files_to_post);
|
|
}
|
|
|
|
|
|
|
|
return(CE_STATUS);
|
|
}
|
|
|
|
/* parse_http_mime_headers
|
|
*
|
|
* parses the mime headers as they are received from the
|
|
* HTTP server.
|
|
*
|
|
* Returns the TCP status code
|
|
*
|
|
*/
|
|
|
|
PRIVATE int
|
|
net_parse_http_mime_headers (ActiveEntry *ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
char *line;
|
|
char *value;
|
|
|
|
CE_STATUS = NET_BufferedReadLine(cd->connection->sock, &line, &CD_LINE_BUFFER,
|
|
&CD_LINE_BUFFER_SIZE, &CD_PAUSE_FOR_READ);
|
|
|
|
if(CE_STATUS < 0)
|
|
{
|
|
NET_ExplainErrorDetails(MK_TCP_READ_ERROR, SOCKET_ERRNO);
|
|
|
|
/* return TCP error
|
|
*/
|
|
return MK_TCP_READ_ERROR;
|
|
}
|
|
|
|
if(CE_STATUS == 0)
|
|
{
|
|
/* if this is set just return so we can use
|
|
* the cached copythat
|
|
*/
|
|
if(CD_USE_COPY_FROM_CACHE)
|
|
{
|
|
/* clear the URL content fields so that
|
|
* the 304 object doesn't effect the actual
|
|
* cache file
|
|
*/
|
|
if(!CE_URL_S->preset_content_type)
|
|
PR_FREEIF(CE_URL_S->content_type);
|
|
CE_URL_S->content_type = NULL;
|
|
CE_URL_S->content_length = 0;
|
|
CE_URL_S->real_content_length = 0;
|
|
CE_URL_S->last_modified = 0;
|
|
PR_FREEIF(CE_URL_S->content_encoding);
|
|
CE_URL_S->content_encoding = NULL;
|
|
PR_FREEIF(CE_URL_S->content_name);
|
|
CE_URL_S->content_name = NULL;
|
|
CD_NEXT_STATE = HTTP_DONE;
|
|
CD_PAUSE_FOR_READ = FALSE;
|
|
return(MK_USE_COPY_FROM_CACHE);
|
|
}
|
|
|
|
/* no mo data */
|
|
if(CD_USE_PROXY_TUNNEL && !CD_PROXY_TUNNEL_SETUP_DONE)
|
|
{
|
|
/* we have now successfully initiated a proxy tunnel
|
|
* connection to a remote host
|
|
*
|
|
* now we need to give the file descriptor
|
|
* to the xxx library and let it initiate
|
|
* a secure connection
|
|
* after that we need to send a normal
|
|
* http request
|
|
*
|
|
* ADD STUFF HERE KIPP!!!!
|
|
*
|
|
* fd is cd->connection->sock
|
|
*/
|
|
if(ce->URL_s->files_to_post)
|
|
CD_NEXT_STATE = HTTP_BEGIN_UPLOAD_FILE;
|
|
else
|
|
CD_NEXT_STATE = HTTP_SEND_REQUEST;
|
|
CD_PROXY_TUNNEL_SETUP_DONE = TRUE;
|
|
}
|
|
else if(ce->URL_s->files_to_post)
|
|
{
|
|
if(ce->URL_s->files_to_post[0]
|
|
&& ce->URL_s->server_status/100 == 2)
|
|
{
|
|
if(ce->URL_s->can_reuse_connection)
|
|
{
|
|
CD_NEXT_STATE = HTTP_BEGIN_UPLOAD_FILE;
|
|
}
|
|
else
|
|
{
|
|
NET_ClearReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
NET_TotalNumberOfOpenConnections--;
|
|
|
|
/* close the old connection
|
|
*/
|
|
PR_Close(cd->connection->sock);
|
|
cd->connection->sock = NULL;
|
|
|
|
CD_NEXT_STATE = HTTP_START_CONNECT;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CD_NEXT_STATE = HTTP_DONE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CD_NEXT_STATE = HTTP_SETUP_STREAM;
|
|
}
|
|
CD_PAUSE_FOR_READ = FALSE;
|
|
return(0);
|
|
}
|
|
|
|
if(!line)
|
|
return(0); /* wait for next line */
|
|
|
|
TRACEMSG(("parse_http_mime_headers: Parsing headers, got line: %s",line));
|
|
|
|
if(CD_ACTING_AS_PROXY)
|
|
{
|
|
/* save all the headers so we can pass them to the client
|
|
* when neccessary
|
|
*/
|
|
StrAllocCat(CD_SERVER_HEADERS, line);
|
|
StrAllocCat(CD_SERVER_HEADERS, "\r\n"); /* add the \n back on */
|
|
}
|
|
|
|
/* check for end of MIME headers */
|
|
if(*line == '\0' || *line == '\r')
|
|
{
|
|
TRACEMSG(("Finished parsing MIME headers"));
|
|
|
|
CD_PAUSE_FOR_READ = FALSE;
|
|
|
|
HG07606
|
|
if(ce->URL_s->files_to_post
|
|
&& ce->URL_s->server_status/100 == 2)
|
|
{
|
|
if(ce->URL_s->files_to_post[0])
|
|
{
|
|
if(ce->URL_s->can_reuse_connection)
|
|
{
|
|
CD_NEXT_STATE = HTTP_BEGIN_UPLOAD_FILE;
|
|
}
|
|
else
|
|
{
|
|
NET_ClearReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
NET_TotalNumberOfOpenConnections--;
|
|
|
|
/* close the old connection
|
|
*/
|
|
PR_Close(cd->connection->sock);
|
|
cd->connection->sock = NULL;
|
|
|
|
CD_NEXT_STATE = HTTP_START_CONNECT;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CD_NEXT_STATE = HTTP_DONE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CD_NEXT_STATE = HTTP_SETUP_STREAM;
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
value = PL_strchr(line, ':');
|
|
if(value)
|
|
value++;
|
|
NET_ParseMimeHeader(CE_FORMAT_OUT, CE_WINDOW_ID, CE_URL_S, line, value, TRUE);
|
|
|
|
return(1);
|
|
}
|
|
|
|
/* setup HTTP/1.1 specific protocol defaults */
|
|
PRIVATE void
|
|
net_setup_http11_defaults(ActiveEntry *ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
|
|
ce->URL_s->can_reuse_connection = TRUE;
|
|
|
|
return;
|
|
}
|
|
|
|
/* parses the first line of the http server's response
|
|
* and determines if it is an http1 or http/.9 server
|
|
*
|
|
* returns the tcp status code
|
|
*/
|
|
PRIVATE int
|
|
net_parse_first_http_line (ActiveEntry *ce)
|
|
{
|
|
char *line_feed=0;
|
|
char *ptr;
|
|
int line_size;
|
|
int num_fields;
|
|
char server_version[36];
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
char small_buf[MAX_FIRST_LINE_SIZE+16];
|
|
Bool do_redirect = TRUE;
|
|
|
|
TRACEMSG(("Entered: net_parse_first_http_line"));
|
|
|
|
CE_STATUS = PR_Read(cd->connection->sock, small_buf, MAX_FIRST_LINE_SIZE+10);
|
|
|
|
if(CE_STATUS == 0)
|
|
{
|
|
/* the server dropped the connection? */
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(MK_ZERO_LENGTH_FILE);
|
|
CD_NEXT_STATE = HTTP_ERROR_DONE;
|
|
CD_PAUSE_FOR_READ = FALSE;
|
|
return(MK_ZERO_LENGTH_FILE);
|
|
}
|
|
else if(CE_STATUS < 0)
|
|
{
|
|
int s_error = PR_GetError();
|
|
if (s_error == PR_WOULD_BLOCK_ERROR)
|
|
{
|
|
CD_PAUSE_FOR_READ = TRUE;
|
|
return(0);
|
|
}
|
|
else
|
|
{
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(MK_TCP_READ_ERROR, s_error);
|
|
|
|
/* return TCP error
|
|
*/
|
|
return MK_TCP_READ_ERROR;
|
|
}
|
|
}
|
|
|
|
/* this is where kipp says that I can finally query
|
|
* for the security data. We can't do it after the
|
|
* connect since the handshake isn't done yet...
|
|
*/
|
|
/* clear existing data
|
|
*/
|
|
PR_FREEIF(CE_URL_S->sec_info);
|
|
ce->URL_s->sec_info = SECNAV_SSLSocketStatus(cd->connection->sock,
|
|
&ce->URL_s->security_on);
|
|
|
|
#ifdef MOZILLA_CLIENT
|
|
if ( ce->URL_s->redirect_sec_info &&
|
|
( ! (CD_USE_PROXY_TUNNEL && !CD_PROXY_TUNNEL_SETUP_DONE) ) ) {
|
|
/* don't do the redirect check when talking to the proxy,
|
|
* wait for it to come through here a second time, when
|
|
* we are really using xxx to talk to the real server.
|
|
*/
|
|
|
|
PRBool compare;
|
|
|
|
compare = SECNAV_CompareCertsForRedirection(ce->URL_s->sec_info,
|
|
ce->URL_s->redirect_sec_info);
|
|
|
|
/* now that we are done with the redirecting info destroy it */
|
|
PR_Free(ce->URL_s->redirect_sec_info);
|
|
ce->URL_s->redirect_sec_info = NULL;
|
|
|
|
if ( compare != PR_TRUE ) {
|
|
/* certs are different */
|
|
|
|
do_redirect = (Bool)SECNAV_SecurityDialog(CE_WINDOW_ID,
|
|
SD_REDIRECTION_TO_SECURE_SITE);
|
|
if ( !do_redirect ) {
|
|
/* stop connection here!! */
|
|
CE_URL_S->error_msg = 0; /* XXX - is this right? */
|
|
|
|
/* return TCP error
|
|
*/
|
|
return MK_TCP_READ_ERROR;
|
|
}
|
|
}
|
|
}
|
|
#endif /* MOZILLA_CLIENT */
|
|
|
|
/* CE_STATUS greater than 0
|
|
*/
|
|
BlockAllocCat(CD_LINE_BUFFER, CD_LINE_BUFFER_SIZE, small_buf, CE_STATUS);
|
|
CD_LINE_BUFFER_SIZE += CE_STATUS;
|
|
|
|
for(ptr = CD_LINE_BUFFER, line_size=0; line_size < CD_LINE_BUFFER_SIZE; ptr++, line_size++)
|
|
if(*ptr == LF)
|
|
{
|
|
line_feed = ptr;
|
|
break;
|
|
}
|
|
|
|
/* assume HTTP/.9 until we know better
|
|
*/
|
|
cd->protocol_version = POINT_NINE;
|
|
|
|
if(line_feed)
|
|
{
|
|
*server_version = 0;
|
|
|
|
*line_feed = '\0';
|
|
|
|
num_fields = sscanf(CD_LINE_BUFFER, "%20s %d", server_version, &CE_URL_S->server_status);
|
|
|
|
TRACEMSG(("HTTP: Scanned %d fields from first_line: %s", num_fields, CD_LINE_BUFFER));
|
|
|
|
/* Try and make sure this is an HTTP/1.0 reply
|
|
*/
|
|
if (num_fields == 2 || !PL_strncmp("HTTP/", server_version, 5))
|
|
{
|
|
double ver = atof(server_version+5);
|
|
|
|
if(ver > 1.0)
|
|
{
|
|
/* HTTP1.1 */
|
|
cd->protocol_version = ONE_POINT_ONE;
|
|
|
|
net_setup_http11_defaults(ce);
|
|
|
|
PR_ASSERT(ver == 1.1);
|
|
}
|
|
else
|
|
{
|
|
/* HTTP1 */
|
|
cd->protocol_version = ONE_POINT_O;
|
|
|
|
PR_ASSERT(ver == 1.0 || ver == 0.0); /* allow 0 bug */
|
|
}
|
|
}
|
|
|
|
/* put the line back the way it should be
|
|
*/
|
|
*line_feed = LF;
|
|
}
|
|
else if(CD_LINE_BUFFER_SIZE < MAX_FIRST_LINE_SIZE)
|
|
{
|
|
return(0); /* not ready to process */
|
|
}
|
|
|
|
if(cd->connection->prev_cache && cd->protocol_version == POINT_NINE)
|
|
{
|
|
/* got a non HTTP/1.0 or above response from
|
|
* server that must support HTTP/1.0 since it
|
|
* supports keep-alive. The connection
|
|
* must be in a bad state now so
|
|
* restart the whole thang by going to the ERROR state
|
|
*/
|
|
CD_NEXT_STATE = HTTP_ERROR_DONE;
|
|
}
|
|
|
|
/* if we are getting a successful read here
|
|
* then the connection is valid
|
|
*/
|
|
cd->connection_is_valid = TRUE;
|
|
|
|
if(cd->protocol_version == POINT_NINE)
|
|
{ /* HTTP/0.9 */
|
|
NET_cinfo * ctype;
|
|
|
|
TRACEMSG(("Receiving HTTP/0.9"));
|
|
|
|
CE_URL_S->content_length = 0;
|
|
CE_URL_S->real_content_length = 0;
|
|
PR_FREEIF(CE_URL_S->content_encoding);
|
|
CE_URL_S->content_encoding = NULL;
|
|
PR_FREEIF(CE_URL_S->content_name);
|
|
CE_URL_S->content_name = NULL;
|
|
|
|
if(!CE_URL_S->preset_content_type)
|
|
{
|
|
PR_FREEIF(CE_URL_S->content_type);
|
|
CE_URL_S->content_type = NULL;
|
|
|
|
/* fake the content_type since we can't get it */
|
|
ctype = NET_cinfo_find_type(CE_URL_S->address);
|
|
|
|
/* treat unknown types as HTML
|
|
*/
|
|
if(ctype->is_default)
|
|
StrAllocCopy(CE_URL_S->content_type, TEXT_HTML);
|
|
else
|
|
StrAllocCopy(CE_URL_S->content_type, ctype->type);
|
|
|
|
/* fake the content_encoding since we can't get it */
|
|
StrAllocCopy(CE_URL_S->content_encoding,
|
|
(NET_cinfo_find_enc(CE_URL_S->address))->encoding);
|
|
}
|
|
|
|
CD_NEXT_STATE = HTTP_SETUP_STREAM;
|
|
|
|
}
|
|
else
|
|
{
|
|
/* Decode full HTTP/1.0 or 1.1 response */
|
|
|
|
TRACEMSG(("Receiving HTTP1 reply, status: %d", CE_URL_S->server_status));
|
|
|
|
if(CE_URL_S->server_status != 304
|
|
&& (!CD_USE_PROXY_TUNNEL || CD_PROXY_TUNNEL_SETUP_DONE))
|
|
{
|
|
/* if we don't have a 304, zero all
|
|
* the content headers so that they
|
|
* don't interfere with the
|
|
* incoming object
|
|
*
|
|
* don't zero these in the case of a proxy tunnel
|
|
* since this isn't the real request this
|
|
* is just the connection open request response
|
|
*/
|
|
if(!CE_URL_S->preset_content_type)
|
|
{
|
|
PR_FREEIF(CE_URL_S->content_type);
|
|
CE_URL_S->content_type = NULL;
|
|
}
|
|
CE_URL_S->content_length = 0;
|
|
CE_URL_S->real_content_length = 0;
|
|
PR_FREEIF(CE_URL_S->content_encoding);
|
|
CE_URL_S->content_encoding = NULL;
|
|
PR_FREEIF(CE_URL_S->content_name);
|
|
CE_URL_S->content_name = NULL;
|
|
}
|
|
|
|
switch (CE_URL_S->server_status / 100)
|
|
{
|
|
case 1:
|
|
if(CE_URL_S->server_status == 100)
|
|
{
|
|
char * end_of_line;
|
|
int cur_line_size;
|
|
|
|
/* 100 continue. Ignore line and continue looking
|
|
* for first line
|
|
*/
|
|
|
|
/* strip the "HTTP/1.1 100 Continue" from the line buffer */
|
|
end_of_line = line_feed;
|
|
cur_line_size = (end_of_line - cd->line_buffer)+1;
|
|
|
|
/* absorb any blank lines after the 100 continue */
|
|
while(cd->line_buffer_size > cur_line_size)
|
|
{
|
|
if(line_feed[1] == CR || line_feed[1] == LF)
|
|
{
|
|
cur_line_size++;
|
|
end_of_line++;
|
|
}
|
|
else
|
|
{
|
|
/* anything else stop */
|
|
break;
|
|
}
|
|
}
|
|
|
|
cd->line_buffer_size -= cur_line_size;
|
|
if(cd->line_buffer_size)
|
|
memmove(cd->line_buffer, end_of_line+1, cd->line_buffer_size);
|
|
|
|
/* by not setting CD_NEXT_STATE to something different
|
|
* we will come back to this function and look for another
|
|
* first line.
|
|
*/
|
|
return(0);
|
|
|
|
}
|
|
break;
|
|
|
|
case 2: /* Succesful reply */
|
|
|
|
/* Since we
|
|
* are getting a new copy, delete the old one
|
|
* from the cache
|
|
*/
|
|
#ifdef MOZILLA_CLIENT
|
|
NET_RemoveURLFromCache(CE_URL_S);
|
|
#endif
|
|
|
|
if((CE_URL_S->server_status == 204
|
|
|| CE_URL_S->server_status == 201)
|
|
&& !CD_ACTING_AS_PROXY)
|
|
{
|
|
if(ce->URL_s->files_to_post && ce->URL_s->files_to_post[0])
|
|
{
|
|
/* fall through since we need to get
|
|
* the headers to complete the
|
|
* file transfer
|
|
*/
|
|
}
|
|
else
|
|
{
|
|
CE_STATUS = MK_NO_ACTION;
|
|
CD_NEXT_STATE = HTTP_ERROR_DONE;
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
}
|
|
else if(cd->partial_cache_file
|
|
&& ce->URL_s->range_header
|
|
&& CE_URL_S->server_status != 206)
|
|
{
|
|
/* we asked for a range and got back
|
|
* the whole document. Something
|
|
* went wrong, error out.
|
|
*/
|
|
CE_STATUS = MK_TCP_READ_ERROR;
|
|
ce->URL_s->error_msg = NET_ExplainErrorDetails(
|
|
MK_TCP_READ_ERROR,
|
|
XP_ERRNO_EIO);
|
|
CD_NEXT_STATE = HTTP_ERROR_DONE;
|
|
return(CE_STATUS);
|
|
}
|
|
|
|
break;
|
|
|
|
case 3: /* redirection and other such stuff */
|
|
|
|
if(!CD_ACTING_AS_PROXY
|
|
&& (CE_URL_S->server_status == 301 || CE_URL_S->server_status == 302))
|
|
{
|
|
/* Redirect with GET only (change POST to get)
|
|
*
|
|
* Supported within the HTTP module.
|
|
* Will retry after mime parsing
|
|
*/
|
|
TRACEMSG(("Got Redirect code"));
|
|
CD_DOING_REDIRECT = TRUE;
|
|
}
|
|
if(!CD_ACTING_AS_PROXY && CE_URL_S->server_status == 307)
|
|
{
|
|
/* Redirect without changing METHOD
|
|
*
|
|
* Supported within the HTTP module.
|
|
* Will retry after mime parsing
|
|
*/
|
|
TRACEMSG(("Got Redirect code"));
|
|
CD_DOING_REDIRECT = TRUE;
|
|
cd->save_redirect_method = TRUE;
|
|
}
|
|
else if(CE_URL_S->server_status == 304)
|
|
{
|
|
/* use the copy from the cache since it wasn't modified
|
|
*
|
|
* note: this will work with proxy clients too
|
|
*/
|
|
if(CE_URL_S->last_modified)
|
|
{
|
|
#ifdef MOZILLA_CLIENT
|
|
/* check to see if we just now entered a secure space
|
|
*
|
|
* don't do if this is coming from history
|
|
* don't do this if about to redirect
|
|
*/
|
|
if(CE_URL_S->security_on
|
|
&& (CE_FORMAT_OUT == FO_CACHE_AND_PRESENT
|
|
|| CE_FORMAT_OUT == FO_PRESENT)
|
|
&& !CE_URL_S->history_num)
|
|
{
|
|
History_entry * h = SHIST_GetCurrent(&CE_WINDOW_ID->hist);
|
|
|
|
if(!h || !h->security_on)
|
|
SECNAV_SecurityDialog(CE_WINDOW_ID,
|
|
SD_ENTERING_SECURE_SPACE);
|
|
}
|
|
#endif /* MOZILLA_CLIENT */
|
|
|
|
CD_USE_COPY_FROM_CACHE = TRUE;
|
|
/* no longer return since we need to parse
|
|
* headers from the server
|
|
*
|
|
* return(MK_USE_COPY_FROM_CACHE);
|
|
*/
|
|
}
|
|
|
|
/* else continue since the server messed up
|
|
* and sent us a 304 even without us having sent
|
|
* it an if-modified-since header
|
|
*/
|
|
}
|
|
break;
|
|
|
|
case 4: /* client error */
|
|
|
|
CE_URL_S->preset_content_type = FALSE;
|
|
|
|
if(CE_URL_S->server_status == 401 && !CD_ACTING_AS_PROXY)
|
|
{
|
|
/* never do this if acting as a proxy
|
|
* If we are a proxy then just pass on
|
|
* the headers and the document.
|
|
*
|
|
* if authorization_required is set we will check
|
|
* below after parsing the MIME headers to see
|
|
* if we should redo the request with an authorization
|
|
* string
|
|
*/
|
|
CD_AUTH_REQUIRED = TRUE;
|
|
|
|
/* Since we
|
|
* are getting a new copy, delete the old one
|
|
* from the cache
|
|
*/
|
|
#ifdef MOZILLA_CLIENT
|
|
NET_RemoveURLFromCache(CE_URL_S);
|
|
#endif
|
|
|
|
/* we probably want to cache this unless
|
|
* the user chooses not to authorize himself
|
|
*/
|
|
}
|
|
else if (CE_URL_S->server_status == 407 && !CD_ACTING_AS_PROXY)
|
|
{
|
|
/* This happens only if acting as a client */
|
|
CD_PROXY_AUTH_REQUIRED = TRUE;
|
|
|
|
/* we probably want to cache this unless
|
|
* the user chooses not to authorize himself
|
|
*/
|
|
}
|
|
else
|
|
{
|
|
/* don't cache unless we have a succesful reply
|
|
*/
|
|
TRACEMSG(("Server did not return success: NOT CACHEING!!"));
|
|
CE_URL_S->dont_cache = TRUE;
|
|
|
|
/* all other codes should be displayed */
|
|
}
|
|
break;
|
|
|
|
case 5: /* server error code */
|
|
TRACEMSG(("Server did not return success: NOT CACHEING!!!"));
|
|
CE_URL_S->dont_cache = TRUE;
|
|
CE_URL_S->preset_content_type = FALSE;
|
|
#ifdef DO_503
|
|
if(CE_URL_S->server_status == 503 && !CD_ACTING_AS_PROXY)
|
|
{
|
|
CD_SERVER_BUSY_RETRY = TRUE;
|
|
}
|
|
#endif /* DO_503 */
|
|
|
|
/* display the error results */
|
|
break;
|
|
|
|
default: /* unexpected reply code */
|
|
{
|
|
char message_buffer[256];
|
|
sprintf(message_buffer,
|
|
XP_GetString(XP_ALERT_UNKNOWN_STATUS),
|
|
CE_URL_S->server_status);
|
|
FE_Alert(CE_WINDOW_ID, message_buffer);
|
|
|
|
/* don't cache unless we have a succesful reply
|
|
*/
|
|
TRACEMSG(("Server did not return 200: NOT CACHEING!!!"));
|
|
CE_URL_S->dont_cache = TRUE;
|
|
}
|
|
break;
|
|
} /* Switch on server_status/100 */
|
|
|
|
CD_NEXT_STATE = HTTP_PARSE_MIME_HEADERS;
|
|
|
|
} /* Full HTTP reply */
|
|
|
|
return(0); /* good */
|
|
|
|
}
|
|
|
|
/* if we were posting a file and received an error put the
|
|
* current file we were posting back on the end of the list
|
|
* so that our state is correctly reset
|
|
*/
|
|
PRIVATE void
|
|
net_revert_post_data(ActiveEntry * ce)
|
|
{
|
|
if(ce->URL_s->files_to_post && ce->URL_s->post_data)
|
|
{
|
|
/* find the end of the files to post array */
|
|
int index=0;
|
|
|
|
for(; ce->URL_s->files_to_post[index]; index++)
|
|
; /* null body */
|
|
|
|
/* this will not explode even if the malloc fails */
|
|
ce->URL_s->files_to_post[index] = PL_strdup(ce->URL_s->post_data);
|
|
|
|
ce->URL_s->files_to_post[index+1] = NULL;
|
|
|
|
if(ce->URL_s->post_to)
|
|
{
|
|
ce->URL_s->post_to[index] = PL_strdup(ce->URL_s->address);
|
|
|
|
ce->URL_s->post_to[index+1] = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* sets up the stream and performs special actions like redirect and
|
|
* retry on authorization
|
|
*
|
|
* returns the tcp status code
|
|
*/
|
|
PRIVATE int
|
|
net_setup_http_stream(ActiveEntry * ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
XP_Bool need_to_do_again = FALSE;
|
|
MWContext * stream_context;
|
|
|
|
TRACEMSG(("NET_ProcessHTTP: setting up stream"));
|
|
|
|
/* save this since it can be changed in lots
|
|
* of places. This will be used for graph progress
|
|
* and to terminate the tranfer
|
|
*/
|
|
if(!ce->URL_s->high_range)
|
|
cd->original_content_length = CE_URL_S->content_length;
|
|
else
|
|
cd->original_content_length = ce->URL_s->high_range
|
|
- ce->URL_s->low_range
|
|
+ 1;
|
|
|
|
if ((ce->URL_s->method == URL_HEAD_METHOD) &&
|
|
( CD_AUTH_REQUIRED == FALSE))
|
|
{
|
|
/* We only wanted the head, so we should stop doing anything else. */
|
|
CD_NEXT_STATE = HTTP_DONE;
|
|
return 0;
|
|
}
|
|
|
|
/* if this is set just return so that we can use
|
|
* the cached copy
|
|
*/
|
|
if(CD_USE_COPY_FROM_CACHE)
|
|
{
|
|
/* if this is a partial cache file situation
|
|
* then we will keep going and switch later
|
|
* on in this file.
|
|
*
|
|
* If it's not a partial cache file then
|
|
* leave the HTTP module and go to the
|
|
* file module to display the file
|
|
*/
|
|
if(!cd->partial_cache_file)
|
|
{
|
|
/* clear the URL content fields so that
|
|
* the 304 object doesn't effect the actual
|
|
* cache file
|
|
*/
|
|
if(!CE_URL_S->preset_content_type)
|
|
{
|
|
PR_FREEIF(CE_URL_S->content_type);
|
|
CE_URL_S->content_type = NULL;
|
|
}
|
|
CE_URL_S->content_length = 0;
|
|
CE_URL_S->real_content_length = 0;
|
|
PR_FREEIF(CE_URL_S->content_encoding);
|
|
CE_URL_S->content_encoding = NULL;
|
|
PR_FREEIF(CE_URL_S->content_name);
|
|
CE_URL_S->content_name = NULL;
|
|
CD_NEXT_STATE = HTTP_DONE;
|
|
CD_PAUSE_FOR_READ = FALSE;
|
|
return(MK_USE_COPY_FROM_CACHE);
|
|
}
|
|
else
|
|
{
|
|
/* set the correct content length so that
|
|
* the cache gets it right
|
|
*/
|
|
ce->URL_s->content_length = ce->URL_s->real_content_length;
|
|
}
|
|
}
|
|
|
|
/* do we need to start the tranfer over with authorization?
|
|
*/
|
|
if(CD_AUTH_REQUIRED)
|
|
{
|
|
/* clear to prevent tight loop */
|
|
int status;
|
|
NET_ClearReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
status = NET_AskForAuthString(CE_WINDOW_ID,
|
|
CE_URL_S,
|
|
CE_URL_S->authenticate,
|
|
CE_URL_S->protection_template,
|
|
CD_SENT_AUTHORIZATION);
|
|
|
|
if(status == NET_RETRY_WITH_AUTH)
|
|
need_to_do_again = TRUE;
|
|
else
|
|
CE_URL_S->dont_cache = TRUE;
|
|
|
|
NET_SetReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
}
|
|
#if defined(XP_WIN) && defined(MOZILLA_CLIENT)
|
|
|
|
#define COMPUSERVE_HEADER_NAME "Remote-Passphrase"
|
|
|
|
else if(CE_URL_S->authenticate && !PL_strncasecmp(CE_URL_S->authenticate,
|
|
COMPUSERVE_HEADER_NAME,
|
|
sizeof(COMPUSERVE_HEADER_NAME) - 1))
|
|
{
|
|
/* compuserve auth requires us to send all authenticate
|
|
* headers into their code to verify the authentication
|
|
*/
|
|
int status = WFE_DoCompuserveAuthenticate(CE_WINDOW_ID,
|
|
CE_URL_S,
|
|
CE_URL_S->authenticate);
|
|
|
|
if(status == NET_AUTH_FAILED_DONT_DISPLAY)
|
|
{
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(MK_COMPUSERVE_AUTH_FAILED);
|
|
|
|
return(MK_COMPUSERVE_AUTH_FAILED);
|
|
}
|
|
}
|
|
#endif /* XP_WIN and MOZILLA_CLIENT */
|
|
|
|
if(CD_PROXY_AUTH_REQUIRED)
|
|
{
|
|
/* This was hacked in because proxy auth can be required when the Auto-config url
|
|
* itself requires authorization. We used to ask for proxy auth only when the proxy was
|
|
* input directly into the prefs. Now we check to see if there's a pacurl (if there
|
|
* is, there can't be a proxy url simultaneously) use it, otherwise we're using a
|
|
* proxy from the prefs.
|
|
*/
|
|
const char *tempURL=NULL;
|
|
char *proxyServer=NULL;
|
|
|
|
/* Figure out which kind of proxy we're using: PAC or straight proxy.
|
|
* DON'T FREE tempURL!!!
|
|
*/
|
|
if ( (tempURL = net_GetPACUrl()) && (*tempURL) )
|
|
proxyServer = NET_ParseURL(tempURL, GET_HOST_PART | GET_PATH_PART | GET_USERNAME_PART | GET_PASSWORD_PART);
|
|
else
|
|
proxyServer = CD_PROXY_SERVER;
|
|
|
|
if(NET_AskForProxyAuth(CE_WINDOW_ID,
|
|
proxyServer,
|
|
CE_URL_S->proxy_authenticate,
|
|
CD_SENT_PROXY_AUTH))
|
|
need_to_do_again = TRUE;
|
|
else
|
|
CE_URL_S->dont_cache = TRUE;
|
|
/* Only free the our temp proxy server if it's not pointing to CD_PROXY_SERVER.
|
|
* We don't want to be freeing someone elses memory, we were just temporarily
|
|
* pointing to it.
|
|
*/
|
|
if (tempURL)
|
|
PR_FREEIF(proxyServer);
|
|
}
|
|
|
|
if (need_to_do_again)
|
|
{
|
|
NET_ClearReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
PR_Close(cd->connection->sock);
|
|
NET_TotalNumberOfOpenConnections--;
|
|
cd->connection->sock = NULL;
|
|
CD_SEND_HTTP1 = TRUE;
|
|
CD_AUTH_REQUIRED = FALSE;
|
|
CD_PROXY_AUTH_REQUIRED = FALSE;
|
|
CD_NEXT_STATE = HTTP_START_CONNECT;
|
|
CE_URL_S->last_modified = 0; /* clear this so we don't get 304 loopage */
|
|
|
|
/* we don't know if the connection is valid anymore */
|
|
cd->connection_is_valid = FALSE;
|
|
|
|
/* clear the buffer
|
|
*/
|
|
PR_FREEIF(CD_LINE_BUFFER);
|
|
CD_LINE_BUFFER = NULL;
|
|
CD_LINE_BUFFER_SIZE = 0;
|
|
|
|
/* if necessary */
|
|
PR_FREEIF(CD_SERVER_HEADERS);
|
|
CD_SERVER_HEADERS = NULL;
|
|
|
|
return(0); /* continue */
|
|
}
|
|
else if (CD_DOING_REDIRECT && CE_URL_S->redirecting_url &&
|
|
/* try and prevent a circular loop. wont work for dual doc loop
|
|
*/
|
|
PL_strcmp(CE_URL_S->redirecting_url, CE_URL_S->address))
|
|
{
|
|
Bool do_redirect=TRUE;
|
|
char *curURLHost, *redirectURLHost;
|
|
char *curPort, *redirectPort;
|
|
|
|
#ifdef MOZILLA_CLIENT
|
|
/* update the global history since we wont have the same
|
|
* url later
|
|
*/
|
|
GH_UpdateGlobalHistory(CE_URL_S);
|
|
#endif /* MOZILLA_CLIENT */
|
|
|
|
/* Inserted for security reasons. Ie. java applets being told to redirect to places they shouldn't be.*/
|
|
if(CE_URL_S->dontAllowDiffHostRedirect) {
|
|
|
|
if( !(curURLHost = NET_ParseURL(CE_URL_S->address, GET_HOST_PART)) )
|
|
return MK_INTERRUPTED;
|
|
if( !(redirectURLHost = NET_ParseURL(CE_URL_S->redirecting_url, GET_HOST_PART)) ) {
|
|
PR_Free(curURLHost);
|
|
return MK_INTERRUPTED;
|
|
}
|
|
|
|
if ( (curPort = PL_strchr(curURLHost, ':')) != NULL)
|
|
*curPort='\0';
|
|
if ( (redirectPort = PL_strchr(redirectURLHost, ':')) != NULL)
|
|
*redirectPort='\0';
|
|
|
|
if(PL_strcasecmp(curURLHost, redirectURLHost)) {
|
|
PR_Free(curURLHost);
|
|
PR_Free(redirectURLHost);
|
|
PR_FREEIF(CE_URL_S->redirecting_url);
|
|
CE_URL_S->redirecting_url = NULL;
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(MK_REDIRECT_ATTEMPT_NOT_ALLOWED);
|
|
return MK_REDIRECT_ATTEMPT_NOT_ALLOWED;
|
|
}
|
|
|
|
PR_Free(curURLHost);
|
|
PR_Free(redirectURLHost);
|
|
|
|
} /* End URL_s->dontAllowDiffHostRedirect */
|
|
|
|
if(CE_FORMAT_OUT == FO_CACHE_AND_PRESENT) {
|
|
if(NET_IsURLSecure(CE_URL_S->address)) { /* current URL is secure*/
|
|
if(NET_IsURLSecure(CE_URL_S->redirecting_url)) {
|
|
/* new URL is secure */
|
|
/* save the cert of the guy doing the redirect */
|
|
ce->URL_s->redirect_sec_info =
|
|
SECNAV_CopySSLSocketStatus(ce->URL_s->sec_info);
|
|
|
|
} else { /* new URL is not secure */
|
|
/* ask the user to confirm */
|
|
do_redirect = (Bool)SECNAV_SecurityDialog(CE_WINDOW_ID,
|
|
SD_REDIRECTION_TO_INSECURE_DOC);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* OK, now we've got the redirection URL stored
|
|
* in redirecting_url.
|
|
* Now change the URL in the URL structure and do the whole
|
|
* thing again
|
|
*/
|
|
StrAllocCopy(CE_URL_S->address, CE_URL_S->redirecting_url);
|
|
|
|
/* if we were posting a file and received an error put the
|
|
* current file we were posting back on the end of the list
|
|
* so that our state is correctly reset
|
|
*/
|
|
net_revert_post_data(ce);
|
|
|
|
if(!cd->save_redirect_method)
|
|
{
|
|
PR_FREEIF(CE_URL_S->post_data);
|
|
CE_URL_S->post_data = NULL;
|
|
PR_FREEIF(CE_URL_S->post_headers);
|
|
CE_URL_S->post_headers = NULL;
|
|
CE_URL_S->post_data_size = 0;
|
|
CE_URL_S->method = URL_GET_METHOD;
|
|
}
|
|
|
|
/* clear these */
|
|
if(!CE_URL_S->preset_content_type)
|
|
PR_FREEIF(CE_URL_S->content_type);
|
|
CE_URL_S->content_type = NULL;
|
|
PR_FREEIF(CE_URL_S->content_encoding);
|
|
CE_URL_S->content_encoding = NULL;
|
|
CE_URL_S->content_length = 0; /* reset */
|
|
CE_URL_S->real_content_length = 0; /* reset */
|
|
CE_URL_S->last_modified = 0; /* reset */
|
|
|
|
CD_POSTING = FALSE;
|
|
|
|
CE_URL_S->address_modified = TRUE;
|
|
|
|
if(do_redirect)
|
|
return(MK_DO_REDIRECT); /*fall out of HTTP and load the redirecting url */
|
|
else
|
|
return(MK_INTERRUPTED);
|
|
}
|
|
else if(CD_SERVER_BUSY_RETRY)
|
|
{
|
|
/* the redirecting mechanism works well for the retry
|
|
* since it is really the same thing except the url stays
|
|
* the same
|
|
*/
|
|
return(MK_DO_REDIRECT); /* fall out of HTTP and reload the url */
|
|
}
|
|
|
|
#ifdef MOZILLA_CLIENT
|
|
/* check to see if we just now entered a secure space
|
|
*
|
|
* don't do if this is coming from history
|
|
* don't do this if about to redirect
|
|
*/
|
|
if(CE_URL_S->security_on
|
|
&& (CE_FORMAT_OUT == FO_CACHE_AND_PRESENT || CE_FORMAT_OUT == FO_PRESENT)
|
|
&& !CE_URL_S->history_num)
|
|
{
|
|
History_entry * h = SHIST_GetCurrent(&CE_WINDOW_ID->hist);
|
|
XP_Bool warn = FALSE;
|
|
|
|
if (h == NULL) {
|
|
/* Deal with frames. If the window doesn't have history,
|
|
* then it is a new window or a new frame cell.
|
|
*/
|
|
if ( ce->window_id->grid_parent != NULL ) {
|
|
h = SHIST_GetCurrent(&ce->window_id->grid_parent->hist);
|
|
if ( !h->security_on ) {
|
|
/* parent frame is not secure */
|
|
warn = TRUE;
|
|
}
|
|
} else {
|
|
/* no parent frame - this is a top level window */
|
|
warn = TRUE;
|
|
}
|
|
} else if ( !h->security_on ) {
|
|
warn = TRUE;
|
|
}
|
|
if ( warn ) {
|
|
SECNAV_SecurityDialog(CE_WINDOW_ID, SD_ENTERING_SECURE_SPACE);
|
|
}
|
|
}
|
|
#endif /* MOZILLA_CLIENT */
|
|
|
|
/* set a default content type if one wasn't given
|
|
*/
|
|
if(!CE_URL_S->content_type
|
|
|| !*CE_URL_S->content_type)
|
|
StrAllocCopy(CE_URL_S->content_type, TEXT_HTML);
|
|
|
|
/* If a stream previously exists from a partial cache
|
|
* situation, reuse it
|
|
*/
|
|
if(!CD_STREAM)
|
|
{
|
|
/* clear to prevent tight loop */
|
|
NET_ClearReadSelect(ce->window_id, cd->connection->sock);
|
|
|
|
#ifdef MOZILLA_CLIENT
|
|
/* if the context can't handle HTML then we
|
|
* need to generate an HTML dialog to handle
|
|
* the message
|
|
*/
|
|
if(ce->URL_s->files_to_post && EDT_IS_EDITOR(ce->window_id))
|
|
{
|
|
Chrome chrome_struct;
|
|
|
|
memset(&chrome_struct, 0, sizeof(Chrome));
|
|
|
|
|
|
chrome_struct.is_modal = TRUE;
|
|
chrome_struct.allow_close = TRUE;
|
|
chrome_struct.allow_resize = TRUE;
|
|
chrome_struct.show_scrollbar = TRUE;
|
|
chrome_struct.w_hint = 400;
|
|
chrome_struct.h_hint = 300;
|
|
#ifdef XP_MAC
|
|
/* on Mac, topmost windows are floating windows not dialogs */
|
|
chrome_struct.topmost = FALSE;
|
|
/* disable commands to change to minimal menu bar; */
|
|
/* avoids confusion about which commands are present */
|
|
chrome_struct.disable_commands = TRUE;
|
|
#else
|
|
chrome_struct.topmost = TRUE;
|
|
#endif
|
|
|
|
|
|
stream_context = FE_MakeNewWindow(ce->window_id,
|
|
NULL,
|
|
NULL,
|
|
&chrome_struct);
|
|
if(!stream_context)
|
|
return (MK_OUT_OF_MEMORY);
|
|
|
|
/* zero out the post_data field so that it doesn't get
|
|
* pushed onto the history stack. Otherwise it can
|
|
* get deleted when the history gets cleared
|
|
*/
|
|
PR_FREEIF(ce->URL_s->post_data);
|
|
ce->URL_s->post_data = NULL;
|
|
ce->URL_s->post_data_is_file = FALSE;
|
|
}
|
|
else
|
|
#endif /* MOZILLA_CLIENT */
|
|
{
|
|
stream_context = CE_WINDOW_ID;
|
|
}
|
|
|
|
/* we can get here on server or proxy errors
|
|
* if we proceed to build the stream with post_data
|
|
* set then the file could get deleted by history
|
|
* cleanup code. Make sure we zero the field
|
|
*/
|
|
if(ce->URL_s->files_to_post && ce->URL_s->post_data)
|
|
{
|
|
/* we shoved the file to post into the post data.
|
|
* remove it so the history doesn't get confused
|
|
* and try and delete the file.
|
|
*/
|
|
PR_FREEIF(ce->URL_s->post_data);
|
|
ce->URL_s->post_data = NULL;
|
|
ce->URL_s->post_data_is_file = FALSE;
|
|
}
|
|
|
|
/* Set up the stream stack to handle the body of the message */
|
|
CD_STREAM = NET_StreamBuilder(CE_FORMAT_OUT,
|
|
CE_URL_S,
|
|
stream_context);
|
|
|
|
if (!CD_STREAM)
|
|
{
|
|
CE_STATUS = MK_UNABLE_TO_CONVERT;
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(MK_UNABLE_TO_CONVERT);
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
HG94794
|
|
NET_SetReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
|
|
if(ce->URL_s->files_to_post)
|
|
{
|
|
char * tmp_string = PL_strdup("<h2>Error uploading files</h2><b>The server responded:<b><hr><p>\n");
|
|
|
|
if(tmp_string)
|
|
PUTSTRING(tmp_string);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* check to see if it's a multipart respose.
|
|
* if it is then we need to do some magic to
|
|
* strip the multipart
|
|
*/
|
|
if(!PL_strncasecmp(ce->URL_s->content_type, "multipart", 9))
|
|
{
|
|
/* reset the state to parse_mime_headers to strip
|
|
* the multipart headers off
|
|
*/
|
|
CD_NEXT_STATE = HTTP_PARSE_MIME_HEADERS;
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
}
|
|
|
|
if(CD_USE_COPY_FROM_CACHE)
|
|
{
|
|
/* we can only get here if it's a partial cache file */
|
|
CD_NEXT_STATE = HTTP_BEGIN_PUSH_PARTIAL_CACHE_FILE;
|
|
CD_USE_COPY_FROM_CACHE = FALSE;
|
|
}
|
|
else
|
|
{
|
|
/* start the graph progress indicator
|
|
*/
|
|
FE_GraphProgressInit(CE_WINDOW_ID,
|
|
CE_URL_S,
|
|
cd->original_content_length);
|
|
CD_DESTROY_GRAPH_PROGRESS = TRUE; /* we will need to destroy it */
|
|
|
|
CD_NEXT_STATE = HTTP_PULL_DATA;
|
|
|
|
if(CD_ACTING_AS_PROXY && CD_SERVER_HEADERS)
|
|
{
|
|
CE_STATUS = PUTBLOCK(CD_SERVER_HEADERS,
|
|
PL_strlen(CD_SERVER_HEADERS));
|
|
CD_DISPLAYED_SOME_DATA = TRUE;
|
|
}
|
|
|
|
{
|
|
char * nonProxyHost = NET_ParseURL(CE_URL_S->address, GET_HOST_PART);
|
|
if (nonProxyHost) {
|
|
char* msg = PR_smprintf(XP_GetString(XP_PROGRESS_TRANSFER_DATA),
|
|
nonProxyHost);
|
|
if (msg) {
|
|
NET_Progress(CE_WINDOW_ID, msg);
|
|
PR_Free(msg);
|
|
}
|
|
PR_Free(nonProxyHost);
|
|
}
|
|
}
|
|
|
|
/* Push though buffered data */
|
|
if(CD_LINE_BUFFER_SIZE)
|
|
{
|
|
/* @@@ bug, check return status and only send
|
|
* up to the return value
|
|
*/
|
|
(*CD_STREAM->is_write_ready)(CD_STREAM);
|
|
CE_STATUS = PUTBLOCK(CD_LINE_BUFFER, CD_LINE_BUFFER_SIZE);
|
|
CE_BYTES_RECEIVED = CD_LINE_BUFFER_SIZE;
|
|
FE_GraphProgress(CE_WINDOW_ID,
|
|
CE_URL_S,
|
|
CE_BYTES_RECEIVED,
|
|
CD_LINE_BUFFER_SIZE,
|
|
CD_ORIGINAL_CONTENT_LENGTH);
|
|
CD_DISPLAYED_SOME_DATA = TRUE;
|
|
}
|
|
}
|
|
|
|
PR_FREEIF(CD_LINE_BUFFER);
|
|
CD_LINE_BUFFER = NULL;
|
|
CD_LINE_BUFFER_SIZE=0;
|
|
|
|
/* check to see if we have read the whole object,
|
|
* and finish the transfer if so.
|
|
*/
|
|
if(CE_STATUS > -1
|
|
&& CD_ORIGINAL_CONTENT_LENGTH
|
|
&& CE_BYTES_RECEIVED >= CD_ORIGINAL_CONTENT_LENGTH)
|
|
{
|
|
/* normal end of transfer */
|
|
CE_STATUS = MK_DATA_LOADED;
|
|
CD_NEXT_STATE = HTTP_DONE;
|
|
CD_PAUSE_FOR_READ = FALSE;
|
|
}
|
|
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
|
|
/* begin pushing a partial cache file down the stream
|
|
*/
|
|
PRIVATE int
|
|
net_http_push_partial_cache_file(ActiveEntry *ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
int32 write_ready, status;
|
|
|
|
write_ready = (*cd->stream->is_write_ready)(cd->stream);
|
|
|
|
write_ready = MIN(write_ready, NET_Socket_Buffer_Size);
|
|
|
|
status = XP_FileRead(NET_Socket_Buffer, write_ready, cd->partial_cache_fp);
|
|
|
|
if(status < 0)
|
|
{
|
|
/* @@@ This is the wrong error code
|
|
*/
|
|
ce->URL_s->error_msg = NET_ExplainErrorDetails(MK_TCP_READ_ERROR,
|
|
SOCKET_ERRNO);
|
|
return(MK_TCP_READ_ERROR);
|
|
}
|
|
else if(status == 0)
|
|
{
|
|
/* all done with reading this file
|
|
*/
|
|
NET_ClearFileReadSelect(ce->window_id, XP_Fileno(cd->partial_cache_fp));
|
|
XP_FileClose(cd->partial_cache_fp);
|
|
cd->partial_cache_fp = NULL;
|
|
|
|
/* set these back in preperation for
|
|
* using the http connection again
|
|
*/
|
|
ce->socket = cd->connection->sock;
|
|
ce->local_file = FALSE;
|
|
|
|
/* add a request-range header
|
|
*/
|
|
PR_ASSERT(!ce->URL_s->range_header);
|
|
ce->URL_s->range_header = PR_smprintf("bytes=%ld-",
|
|
cd->partial_needed);
|
|
|
|
/* the byterange part has not been gotten yet */
|
|
ce->URL_s->last_modified = 0;
|
|
|
|
/* we don't know if the connection is valid anymore
|
|
* because we are going to try it again
|
|
*/
|
|
cd->connection_is_valid = FALSE;
|
|
|
|
if(ce->URL_s->can_reuse_connection)
|
|
{
|
|
NET_SetReadSelect(ce->window_id, cd->connection->sock);
|
|
cd->next_state = HTTP_SEND_REQUEST;
|
|
|
|
/* set the connection to be from the connection cache
|
|
* since we have used it once
|
|
*/
|
|
cd->connection->prev_cache = TRUE;
|
|
}
|
|
else
|
|
{
|
|
NET_ClearReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
NET_ClearConnectSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
PR_Close(cd->connection->sock);
|
|
NET_TotalNumberOfOpenConnections--;
|
|
cd->connection->sock = NULL;
|
|
cd->next_state = HTTP_START_CONNECT;
|
|
}
|
|
|
|
cd->reuse_stream = TRUE;
|
|
|
|
return(0);
|
|
}
|
|
|
|
/* else, push the data read up the stream
|
|
*/
|
|
status = (*cd->stream->put_block)(cd->stream,
|
|
NET_Socket_Buffer,
|
|
status);
|
|
|
|
cd->pause_for_read = TRUE;
|
|
|
|
return(status);
|
|
}
|
|
|
|
/* begin pushing a partial cache file down the stream
|
|
*/
|
|
PRIVATE int
|
|
net_http_begin_push_partial_cache_file(ActiveEntry *ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
char *cache_file = ce->URL_s->cache_file;
|
|
XP_File fp;
|
|
|
|
if(!cache_file
|
|
|| NULL == (fp = XP_FileOpen(cache_file, xpCache, XP_FILE_READ_BIN)))
|
|
{
|
|
ce->URL_s->error_msg = NET_ExplainErrorDetails(MK_UNABLE_TO_OPEN_FILE,
|
|
cache_file);
|
|
return(MK_UNABLE_TO_OPEN_FILE);
|
|
}
|
|
|
|
/* set up read select on the file instead of the socket
|
|
*/
|
|
NET_ClearReadSelect(ce->window_id, cd->connection->sock);
|
|
NET_SetFileReadSelect(ce->window_id, XP_Fileno(fp));
|
|
ce->socket = NULL;
|
|
ce->local_file = TRUE;
|
|
|
|
cd->next_state = HTTP_PUSH_PARTIAL_CACHE_FILE;
|
|
|
|
cd->partial_cache_fp = fp;
|
|
|
|
return(net_http_push_partial_cache_file(ce));
|
|
}
|
|
|
|
/* pulls down all the data
|
|
*
|
|
* returns the tcp status code
|
|
*/
|
|
PRIVATE int
|
|
net_pull_http_data(ActiveEntry * ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
unsigned int write_ready, read_size;
|
|
|
|
TRACEMSG(("NET_ProcessHTTP: pulling data"));
|
|
|
|
/* check to see if the stream is ready for writing
|
|
*/
|
|
write_ready = (*CD_STREAM->is_write_ready)(CD_STREAM);
|
|
|
|
if(!write_ready)
|
|
{
|
|
CD_PAUSE_FOR_READ = TRUE;
|
|
return(0); /* wait until we are ready to write */
|
|
}
|
|
else if(write_ready < (unsigned int) NET_Socket_Buffer_Size)
|
|
{
|
|
read_size = write_ready;
|
|
}
|
|
else
|
|
{
|
|
read_size = NET_Socket_Buffer_Size;
|
|
}
|
|
|
|
CE_STATUS = PR_Read(cd->connection->sock, NET_Socket_Buffer, read_size);
|
|
|
|
CD_PAUSE_FOR_READ = TRUE; /* pause for the next read */
|
|
|
|
if(CE_STATUS > 0)
|
|
{
|
|
CE_BYTES_RECEIVED += CE_STATUS;
|
|
FE_GraphProgress(CE_WINDOW_ID,
|
|
CE_URL_S,
|
|
CE_BYTES_RECEIVED,
|
|
CE_STATUS,
|
|
CD_ORIGINAL_CONTENT_LENGTH);
|
|
|
|
CE_STATUS = PUTBLOCK(NET_Socket_Buffer, CE_STATUS); /* ALEKS */
|
|
CD_DISPLAYED_SOME_DATA = TRUE;
|
|
|
|
if(CD_ORIGINAL_CONTENT_LENGTH
|
|
&& CE_BYTES_RECEIVED >= CD_ORIGINAL_CONTENT_LENGTH)
|
|
{
|
|
/* normal end of transfer */
|
|
CE_STATUS = MK_DATA_LOADED;
|
|
CD_NEXT_STATE = HTTP_DONE;
|
|
CD_PAUSE_FOR_READ = FALSE;
|
|
}
|
|
|
|
}
|
|
else if(CE_STATUS == 0)
|
|
{
|
|
/* transfer finished
|
|
*/
|
|
TRACEMSG(("MKHTTP.c: Caught TCP EOF ending stream"));
|
|
|
|
if(!CD_DISPLAYED_SOME_DATA)
|
|
{
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(MK_ZERO_LENGTH_FILE);
|
|
CD_NEXT_STATE = HTTP_ERROR_DONE;
|
|
CD_PAUSE_FOR_READ = FALSE;
|
|
return(MK_ZERO_LENGTH_FILE);
|
|
}
|
|
|
|
/* return the server status instead of data loaded
|
|
*/
|
|
CE_STATUS = MK_DATA_LOADED;
|
|
CD_NEXT_STATE = HTTP_DONE;
|
|
CD_PAUSE_FOR_READ = FALSE;
|
|
}
|
|
else /* error */
|
|
{
|
|
int err = PR_GetError();
|
|
|
|
TRACEMSG(("TCP Error: %d", err));
|
|
|
|
if (err == PR_WOULD_BLOCK_ERROR)
|
|
{
|
|
CD_PAUSE_FOR_READ = TRUE;
|
|
return (0);
|
|
}
|
|
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(MK_TCP_READ_ERROR, err);
|
|
|
|
/* return TCP error
|
|
*/
|
|
return MK_TCP_READ_ERROR;
|
|
}
|
|
|
|
return STATUS(CE_STATUS);
|
|
|
|
}
|
|
|
|
/* begin loading an object from an http host
|
|
*
|
|
* proxy_server If not empty, the name of a host and/or port that will
|
|
* act as a proxy server for the request. If proxy_server
|
|
* is NULL then no proxy server is used
|
|
*
|
|
*/
|
|
PRIVATE int32
|
|
net_HTTPLoad (ActiveEntry * ce)
|
|
{
|
|
/* get memory for Connection Data */
|
|
HTTPConData * cd = PR_NEW(HTTPConData);
|
|
XP_Bool url_is_secure = FALSE;
|
|
char *use_host;
|
|
|
|
ce->con_data = cd;
|
|
if(!ce->con_data)
|
|
{
|
|
CE_STATUS = MK_OUT_OF_MEMORY;
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(MK_OUT_OF_MEMORY);
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
|
|
/* kill any returns in the URL */
|
|
strtok(ce->URL_s->address, "\r");
|
|
strtok(ce->URL_s->address, "\n");
|
|
|
|
/* init */
|
|
memset(cd, 0, sizeof(HTTPConData));
|
|
CD_PROXY_SERVER = ce->proxy_addr;
|
|
CD_PROXY_CONF = ce->proxy_conf;
|
|
CD_SEND_HTTP1 = TRUE;
|
|
|
|
/* set partial_cache_file if the whole file is not
|
|
* cached
|
|
*/
|
|
if(ce->URL_s->content_length < ce->URL_s->real_content_length)
|
|
{
|
|
cd->partial_cache_file = TRUE;
|
|
cd->partial_needed = ce->URL_s->content_length;
|
|
|
|
#ifdef MOZILLA_CLIENT
|
|
/* if this isn't true then partial cacheing is screwed */
|
|
PR_ASSERT(NET_IsPartialCacheFile(ce->URL_s));
|
|
#else
|
|
PR_ASSERT(0);
|
|
#endif /* MOZILLA_CLIENT */
|
|
}
|
|
|
|
if(CD_PROXY_SERVER)
|
|
{
|
|
use_host = PL_strdup(CD_PROXY_SERVER);
|
|
}
|
|
else
|
|
{
|
|
use_host = NET_ParseURL(ce->URL_s->address, GET_HOST_PART);
|
|
if(!PL_strncasecmp(ce->URL_s->address, "https:", 6))
|
|
url_is_secure = TRUE;
|
|
}
|
|
|
|
if(!use_host)
|
|
{
|
|
PR_Free(ce->con_data);
|
|
CE_STATUS = MK_OUT_OF_MEMORY;
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(MK_OUT_OF_MEMORY);
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
|
|
/* if we are useing the files_to_post method
|
|
* make sure that the directory specified in the
|
|
* URL contains a slash at the end, otherwise
|
|
* it wont work
|
|
*/
|
|
if(ce->URL_s->files_to_post)
|
|
{
|
|
int32 end = PL_strlen(ce->URL_s->address)-1;
|
|
XP_StatStruct stat_entry;
|
|
int i;
|
|
|
|
if(ce->URL_s->address[end] != '/')
|
|
StrAllocCat(ce->URL_s->address, "/");
|
|
|
|
/* run through the list of files and
|
|
* gather the total size
|
|
*/
|
|
for(i=0; ce->URL_s->files_to_post[i]; i++)
|
|
if(-1 != XP_Stat(ce->URL_s->files_to_post[i],
|
|
&stat_entry,
|
|
xpFileToPost))
|
|
cd->total_size_of_files_to_post += stat_entry.st_size;
|
|
|
|
/* start the graph progress indicator
|
|
*/
|
|
FE_GraphProgressInit(ce->window_id,
|
|
ce->URL_s,
|
|
cd->total_size_of_files_to_post);
|
|
|
|
CD_DESTROY_GRAPH_PROGRESS = TRUE; /* we will need to destroy it */
|
|
|
|
#ifdef EDITOR
|
|
/* Don't show the dialog if the data is being delivered to a plug-in */
|
|
if ( (CLEAR_CACHE_BIT(ce->format_out) != FO_PLUGIN)
|
|
&& (CLEAR_CACHE_BIT(ce->format_out) != FO_LOCATION_INDEPENDENCE))
|
|
{
|
|
FE_SaveDialogCreate(ce->window_id, i, ED_SAVE_DLG_PUBLISH);
|
|
cd->destroy_file_upload_progress_dialog = TRUE;
|
|
}
|
|
#endif /* EDITOR */
|
|
}
|
|
|
|
/* check for established connection and use it if available
|
|
*/
|
|
if(http_connection_list)
|
|
{
|
|
HTTPConnection * tmp_con;
|
|
XP_List * list_entry = http_connection_list;
|
|
|
|
/* If the url is secure and we are using a proxy server
|
|
* then never use a cached connection
|
|
*/
|
|
if(!cd->use_proxy_tunnel)
|
|
{
|
|
|
|
while((tmp_con = (HTTPConnection *)XP_ListNextObject(list_entry))
|
|
!= NULL)
|
|
{
|
|
/* if the hostnames match up exactly
|
|
* and security matches up.
|
|
* and the connection
|
|
* is not busy at the moment then reuse this connection.
|
|
*/
|
|
if(!PL_strcmp(tmp_con->hostname, use_host)
|
|
&& url_is_secure == tmp_con->secure
|
|
&& !tmp_con->busy)
|
|
{
|
|
cd->connection = tmp_con;
|
|
cd->connection->sock = cd->connection->sock;
|
|
NET_SetReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
CE_SOCK = cd->connection->sock;
|
|
cd->connection->prev_cache = TRUE; /* this was from the cache */
|
|
TRACEMSG(("Found cached HTTP connection: %d", cd->connection->sock));
|
|
|
|
/* reorder the connection in the list to keep most recently
|
|
* used connections at the end
|
|
*/
|
|
XP_ListRemoveObject(http_connection_list, tmp_con);
|
|
XP_ListAddObjectToEnd(http_connection_list, tmp_con);
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* initialize the connection list
|
|
*/
|
|
http_connection_list = XP_ListNew();
|
|
}
|
|
|
|
if(cd->connection)
|
|
{
|
|
if(CD_USE_PROXY_TUNNEL)
|
|
CD_NEXT_STATE = HTTP_SEND_PROXY_TUNNEL_REQUEST;
|
|
else if(ce->URL_s->files_to_post)
|
|
CD_NEXT_STATE = HTTP_BEGIN_UPLOAD_FILE;
|
|
else
|
|
CD_NEXT_STATE = HTTP_SEND_REQUEST;
|
|
|
|
/* set the connection busy
|
|
*/
|
|
cd->connection->busy = TRUE;
|
|
NET_TotalNumberOfOpenConnections++;
|
|
}
|
|
else
|
|
{
|
|
/* build a control connection structure so we
|
|
* can store the data as we go along
|
|
*/
|
|
cd->connection = PR_NEW(HTTPConnection);
|
|
if(!cd->connection)
|
|
{
|
|
CE_STATUS = MK_OUT_OF_MEMORY;
|
|
CE_URL_S->error_msg = NET_ExplainErrorDetails(MK_OUT_OF_MEMORY);
|
|
PR_Free(use_host);
|
|
PR_Free(ce->con_data);
|
|
return(-1);
|
|
}
|
|
memset(cd->connection, 0, sizeof(HTTPConnection));
|
|
|
|
StrAllocCopy(cd->connection->hostname, use_host);
|
|
|
|
cd->connection->secure = url_is_secure;
|
|
|
|
cd->connection->prev_cache = FALSE; /* this wasn't from the cache */
|
|
|
|
cd->connection->sock = NULL;
|
|
|
|
/* add this structure to the connection list even
|
|
* though it's not really valid yet.
|
|
* we will fill it in as we go and if
|
|
* an error occurs will will remove it from the
|
|
* list. No one else will be able to use it since
|
|
* we will mark it busy.
|
|
*/
|
|
XP_ListAddObject(http_connection_list, cd->connection);
|
|
|
|
/* set the connection busy
|
|
*/
|
|
cd->connection->busy = TRUE;
|
|
|
|
/* if the connection list is larger than MAX_CACHED_HTTP_CONNECTIONS
|
|
* trim it down
|
|
*/
|
|
if(XP_ListCount(http_connection_list) > MAX_CACHED_HTTP_CONNECTIONS)
|
|
{
|
|
HTTPConnection *tmp_con;
|
|
XP_List * list_entry = http_connection_list;
|
|
|
|
TRACEMSG(("More than %d cached connections. Deleteing one...",
|
|
MAX_CACHED_HTTP_CONNECTIONS));
|
|
|
|
while((tmp_con = (HTTPConnection *)XP_ListNextObject(list_entry)) != NULL)
|
|
{
|
|
if(!tmp_con->busy)
|
|
{
|
|
|
|
if(!PL_strncasecmp(tmp_con->hostname, "rl.", 3)
|
|
&& PL_strcasestr(tmp_con->hostname+2, ".netscape.com"))
|
|
{
|
|
/* if there is max plus one we are done, else
|
|
* continue on and remove one
|
|
*/
|
|
if(XP_ListCount(http_connection_list)
|
|
== MAX_CACHED_HTTP_CONNECTIONS+1)
|
|
{
|
|
break; /* from while */
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
/* remove the object */
|
|
XP_ListRemoveObject(http_connection_list, tmp_con);
|
|
PR_Close(tmp_con->sock);
|
|
PR_Free(tmp_con->hostname);
|
|
PR_Free(tmp_con);
|
|
break; /* from while */
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
CD_NEXT_STATE = HTTP_START_CONNECT;
|
|
|
|
}
|
|
|
|
PR_Free(use_host);
|
|
|
|
return STATUS (net_ProcessHTTP(ce));
|
|
}
|
|
|
|
|
|
/* NET_process_HTTP will control the state machine that
|
|
* loads an HTTP document
|
|
*
|
|
* returns negative if the transfer is finished or error'd out
|
|
*
|
|
* returns zero or more if the transfer needs to be continued.
|
|
*/
|
|
PRIVATE int32
|
|
net_ProcessHTTP (ActiveEntry *ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
|
|
TRACEMSG(("Entering NET_ProcessHTTP"));
|
|
|
|
CD_PAUSE_FOR_READ = FALSE; /* already paused; reset */
|
|
|
|
while(!CD_PAUSE_FOR_READ)
|
|
{
|
|
|
|
switch(CD_NEXT_STATE)
|
|
{
|
|
case HTTP_START_CONNECT:
|
|
CE_STATUS = net_start_http_connect(ce);
|
|
break;
|
|
|
|
case HTTP_FINISH_CONNECT:
|
|
CE_STATUS = net_finish_http_connect(ce);
|
|
break;
|
|
|
|
case HTTP_SEND_PROXY_TUNNEL_REQUEST:
|
|
/* send proxy tunnel init stuff
|
|
*/
|
|
CE_STATUS = net_send_proxy_tunnel_request(ce);
|
|
break;
|
|
|
|
case HTTP_BEGIN_UPLOAD_FILE:
|
|
/* form a put request */
|
|
ce->status = net_begin_upload_file (ce);
|
|
break;
|
|
HG51096
|
|
case HTTP_SEND_REQUEST:
|
|
/* send HTTP request */
|
|
CE_STATUS = net_send_http_request(ce);
|
|
break;
|
|
|
|
case HTTP_SEND_POST_DATA:
|
|
CE_STATUS = net_http_send_post_data(ce);
|
|
break;
|
|
|
|
case HTTP_PARSE_FIRST_LINE:
|
|
CE_STATUS = net_parse_first_http_line(ce);
|
|
break;
|
|
|
|
case HTTP_PARSE_MIME_HEADERS:
|
|
CE_STATUS = net_parse_http_mime_headers(ce);
|
|
break;
|
|
|
|
case HTTP_SETUP_STREAM:
|
|
CE_STATUS = net_setup_http_stream(ce);
|
|
break;
|
|
|
|
case HTTP_BEGIN_PUSH_PARTIAL_CACHE_FILE:
|
|
ce->status = net_http_begin_push_partial_cache_file(ce);
|
|
break;
|
|
|
|
case HTTP_PUSH_PARTIAL_CACHE_FILE:
|
|
ce->status = net_http_push_partial_cache_file(ce);
|
|
break;
|
|
|
|
case HTTP_PULL_DATA:
|
|
CE_STATUS = net_pull_http_data(ce);
|
|
break;
|
|
|
|
case HTTP_DONE:
|
|
NET_ClearReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
NET_TotalNumberOfOpenConnections--;
|
|
|
|
if(ce->URL_s->can_reuse_connection && !CD_USE_PROXY_TUNNEL)
|
|
{
|
|
cd->connection->busy = FALSE;
|
|
}
|
|
else
|
|
{
|
|
PR_Close(cd->connection->sock);
|
|
|
|
/* remove the connection from the cache list
|
|
* and free the data
|
|
*/
|
|
XP_ListRemoveObject(http_connection_list, cd->connection);
|
|
if(cd->connection)
|
|
{
|
|
PR_FREEIF(cd->connection->hostname);
|
|
PR_Free(cd->connection);
|
|
}
|
|
}
|
|
|
|
#ifdef MOZILLA_CLIENT
|
|
/* make any meta tag changes take effect
|
|
*/
|
|
NET_RefreshCacheFileExpiration(CE_URL_S);
|
|
#endif /* MOZILLA_CLIENT */
|
|
|
|
if(CD_STREAM)
|
|
{
|
|
COMPLETE_STREAM;
|
|
PR_Free(CD_STREAM);
|
|
CD_STREAM = 0;
|
|
}
|
|
CD_NEXT_STATE = HTTP_FREE;
|
|
break;
|
|
|
|
case HTTP_ERROR_DONE:
|
|
if(cd->connection->sock != NULL)
|
|
{
|
|
NET_ClearDNSSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
NET_ClearReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
NET_ClearConnectSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
PR_Close(cd->connection->sock);
|
|
NET_TotalNumberOfOpenConnections--;
|
|
cd->connection->sock = NULL;
|
|
}
|
|
|
|
if(cd->partial_cache_fp)
|
|
{
|
|
NET_ClearFileReadSelect(ce->window_id,
|
|
XP_Fileno(cd->partial_cache_fp));
|
|
XP_FileClose(cd->partial_cache_fp);
|
|
cd->partial_cache_fp = 0;
|
|
}
|
|
|
|
|
|
if(cd->connection->prev_cache
|
|
&& !cd->connection_is_valid
|
|
&& ce->status != MK_INTERRUPTED)
|
|
{
|
|
if(CD_STREAM && !cd->reuse_stream)
|
|
{
|
|
ABORT_STREAM(CE_STATUS);
|
|
PR_Free(CD_STREAM);
|
|
CD_STREAM = 0;
|
|
}
|
|
|
|
/* the connection came from the cache and
|
|
* may have been stale. Try it again
|
|
*/
|
|
/* clear any error message */
|
|
if(ce->URL_s->error_msg)
|
|
{
|
|
PR_Free(ce->URL_s->error_msg);
|
|
ce->URL_s->error_msg = NULL;
|
|
}
|
|
cd->next_state = HTTP_START_CONNECT;
|
|
ce->status = 0;
|
|
|
|
/* we don't know if the connection is valid anymore
|
|
* because we are going to try it again
|
|
*/
|
|
cd->connection_is_valid = FALSE;
|
|
|
|
cd->connection->prev_cache = FALSE;
|
|
|
|
/* if we were posting a file and received an error put the
|
|
* current file we were posting back on the end of the list
|
|
* so that our state is correctly reset
|
|
*/
|
|
net_revert_post_data(ce);
|
|
|
|
}
|
|
else
|
|
{
|
|
CD_NEXT_STATE = HTTP_FREE;
|
|
|
|
if(CD_STREAM)
|
|
{
|
|
ABORT_STREAM(CE_STATUS);
|
|
PR_Free(CD_STREAM);
|
|
CD_STREAM = 0;
|
|
}
|
|
|
|
/* remove the connection from the cache list
|
|
* and free the data
|
|
*/
|
|
XP_ListRemoveObject(http_connection_list, cd->connection);
|
|
PR_FREEIF(cd->connection->hostname);
|
|
PR_Free(cd->connection);
|
|
}
|
|
|
|
break;
|
|
|
|
case HTTP_FREE:
|
|
|
|
/* close the file upload progress. If a stream was created
|
|
* then some sort of HTTP error occured. Send in an error
|
|
* code
|
|
*/
|
|
#ifdef EDITOR
|
|
if(cd->destroy_file_upload_progress_dialog) {
|
|
/* Convert from netlib errors to editor errors. */
|
|
ED_FileError error = ED_ERROR_NONE;
|
|
if ( (ce->URL_s->server_status != 204 && ce->URL_s->server_status != 201 )
|
|
|| ce->status < 0 )
|
|
error = ED_ERROR_PUBLISHING;
|
|
FE_SaveDialogDestroy(ce->window_id, error, ce->URL_s->post_data);
|
|
/* FE_SaveDialogDestroy(ce->window_id, ce->URL_s->server_status != 204 ? -1 : ce->status, ce->URL_s->post_data); */
|
|
}
|
|
#endif /* EDITOR */
|
|
|
|
if(ce->URL_s->files_to_post && ce->URL_s->post_data)
|
|
{
|
|
/* we shoved the file to post into the post data.
|
|
* remove it so the history doesn't get confused
|
|
* and try and delete the file.
|
|
*/
|
|
PR_FREEIF(ce->URL_s->post_data);
|
|
ce->URL_s->post_data = NULL;
|
|
ce->URL_s->post_data_is_file = FALSE;
|
|
}
|
|
|
|
if(CD_DESTROY_GRAPH_PROGRESS)
|
|
FE_GraphProgressDestroy(CE_WINDOW_ID,
|
|
CE_URL_S,
|
|
CD_ORIGINAL_CONTENT_LENGTH,
|
|
CE_BYTES_RECEIVED);
|
|
|
|
PR_FREEIF(CD_LINE_BUFFER);
|
|
PR_FREEIF(CD_STREAM); /* don't forget the stream */
|
|
PR_FREEIF(CD_SERVER_HEADERS);
|
|
PR_FREEIF(cd->orig_host);
|
|
if(CD_TCP_CON_DATA)
|
|
NET_FreeTCPConData(CD_TCP_CON_DATA);
|
|
PR_FREEIF(cd);
|
|
JSCF_Cleanup();
|
|
return STATUS (-1); /* final end */
|
|
|
|
default: /* should never happen !!! */
|
|
TRACEMSG(("HTTP: BAD STATE!"));
|
|
CD_NEXT_STATE = HTTP_ERROR_DONE;
|
|
break;
|
|
}
|
|
|
|
/* check for errors during load and call error
|
|
* state if found
|
|
*/
|
|
if(CE_STATUS < 0
|
|
&& CE_STATUS != MK_USE_COPY_FROM_CACHE
|
|
&& CD_NEXT_STATE != HTTP_FREE)
|
|
{
|
|
|
|
if (CE_STATUS == MK_MULTIPART_MESSAGE_COMPLETED)
|
|
{
|
|
/* We found the end of a multipart/mixed response
|
|
* from a CGI script in a http keep-alive response
|
|
* That signifies the end of a message.
|
|
*/
|
|
TRACEMSG(("mkhttp.c: End of multipart keep-alive response"));
|
|
|
|
CE_STATUS = MK_DATA_LOADED;
|
|
CD_NEXT_STATE = HTTP_DONE;
|
|
|
|
}
|
|
else if (CE_STATUS == MK_HTTP_TYPE_CONFLICT
|
|
/* Don't retry if were HTTP/.9 */
|
|
&& !CD_SEND_HTTP1
|
|
/* Don't retry if we're posting. */
|
|
&& !CD_POSTING)
|
|
{
|
|
/* Could be a HTTP 0/1 compability problem. */
|
|
TRACEMSG(("HTTP: Read error trying again with HTTP0 request."));
|
|
NET_Progress (CE_WINDOW_ID, XP_GetString(XP_PROGRESS_TRYAGAIN));
|
|
|
|
NET_ClearReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
NET_ClearConnectSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
#ifdef XP_WIN
|
|
if(cd->calling_netlib_all_the_time)
|
|
{
|
|
NET_ClearCallNetlibAllTheTime(CE_WINDOW_ID, "mkhttp");
|
|
}
|
|
#endif /* XP_WIN */
|
|
NET_ClearDNSSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
PR_Close(cd->connection->sock);
|
|
NET_TotalNumberOfOpenConnections--;
|
|
cd->connection->sock = NULL;
|
|
|
|
if(CD_STREAM)
|
|
(*CD_STREAM->abort) (CD_STREAM, CE_STATUS);
|
|
CD_SEND_HTTP1 = FALSE;
|
|
/* go back and send an HTTP0 request */
|
|
CD_NEXT_STATE = HTTP_START_CONNECT;
|
|
}
|
|
else
|
|
{
|
|
CD_NEXT_STATE = HTTP_ERROR_DONE;
|
|
}
|
|
/* don't exit! loop around again and do the free case */
|
|
CD_PAUSE_FOR_READ = FALSE;
|
|
}
|
|
} /* while(!CD_PAUSE_FOR_READ) */
|
|
|
|
return STATUS(CE_STATUS);
|
|
}
|
|
|
|
|
|
/* abort the connection in progress
|
|
*/
|
|
PRIVATE int32
|
|
net_InterruptHTTP(ActiveEntry * ce)
|
|
{
|
|
HTTPConData * cd = (HTTPConData *)ce->con_data;
|
|
|
|
/* if we are currently pulling data and the data is
|
|
* Textual then truncate the file, leave notification and
|
|
* end normally
|
|
*/
|
|
if(CD_NEXT_STATE == HTTP_PULL_DATA
|
|
&& CE_URL_S->content_type
|
|
&& !PL_strcasecmp(CE_URL_S->content_type, TEXT_HTML))
|
|
{
|
|
char buffer[127];
|
|
|
|
if(!PL_strcasecmp(CE_URL_S->content_type, TEXT_HTML))
|
|
PR_snprintf(buffer, sizeof(buffer),
|
|
XP_GetString(XP_HR_TRANSFER_INTERRUPTED));
|
|
else
|
|
PR_snprintf(buffer, sizeof(buffer),
|
|
XP_GetString(XP_TRANSFER_INTERRUPTED));
|
|
|
|
PUTSTRING(buffer);
|
|
|
|
/* steps from HTTP_DONE duplicated, with the addition of
|
|
* NET_RefreshCacheFileExpiration
|
|
*/
|
|
NET_ClearReadSelect(CE_WINDOW_ID, cd->connection->sock);
|
|
PR_Close(cd->connection->sock);
|
|
NET_TotalNumberOfOpenConnections--;
|
|
ABORT_STREAM(MK_INTERRUPTED);
|
|
PR_Free(CD_STREAM);
|
|
CD_STREAM = 0;
|
|
|
|
CE_URL_S->last_modified = 0;
|
|
#ifdef MOZILLA_CLIENT
|
|
/* to make the last_modified change take effect
|
|
*/
|
|
NET_RefreshCacheFileExpiration(CE_URL_S);
|
|
#endif /* MOZILLA_CLIENT */
|
|
|
|
CD_NEXT_STATE = HTTP_FREE;
|
|
|
|
CE_STATUS = MK_DATA_LOADED;
|
|
}
|
|
else
|
|
{
|
|
CD_NEXT_STATE = HTTP_ERROR_DONE;
|
|
CE_STATUS = MK_INTERRUPTED;
|
|
}
|
|
|
|
return STATUS (net_ProcessHTTP(ce));
|
|
}
|
|
|
|
/* Free any memory that might be used in caching etc.
|
|
*/
|
|
PRIVATE void
|
|
net_CleanupHTTP(void)
|
|
{
|
|
/* nothing so far needs freeing */
|
|
return;
|
|
}
|
|
|
|
#define REFERER_HEADER_PREF "network.sendRefererHeader"
|
|
|
|
PRIVATE void
|
|
HTTP_ReadPrefs(void)
|
|
{
|
|
XP_Bool b;
|
|
PREF_GetBoolPref("network.sendRefererHeader", &b);
|
|
NET_SetSendRefererHeader(b);
|
|
}
|
|
|
|
MODULE_PRIVATE int PR_CALLBACK HTTP_PrefChangedFunc(const char *pref, void *data)
|
|
{
|
|
HTTP_ReadPrefs();
|
|
return TRUE;
|
|
}
|
|
|
|
PRIVATE void
|
|
HTTP_InitPrefs(void)
|
|
{
|
|
HTTP_ReadPrefs();
|
|
PREF_RegisterCallback(REFERER_HEADER_PREF ,HTTP_PrefChangedFunc,NULL);
|
|
}
|
|
|
|
MODULE_PRIVATE void
|
|
NET_InitHTTPProtocol(void)
|
|
{
|
|
static NET_ProtoImpl http_proto_impl;
|
|
|
|
HTTP_InitPrefs();
|
|
|
|
http_proto_impl.init = net_HTTPLoad;
|
|
http_proto_impl.process = net_ProcessHTTP;
|
|
http_proto_impl.interrupt = net_InterruptHTTP;
|
|
http_proto_impl.cleanup = net_CleanupHTTP;
|
|
|
|
NET_RegisterProtocolImplementation(&http_proto_impl, HTTP_TYPE_URL);
|
|
NET_RegisterProtocolImplementation(&http_proto_impl, SECURE_HTTP_TYPE_URL);
|
|
|
|
}
|
|
|
|
#ifdef PROFILE
|
|
#pragma profile off
|
|
#endif
|