/* * Copyright 2008 Hans Leidekker for CodeWeavers * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #ifndef _WINE_WINHTTP_PRIVATE_H_ #define _WINE_WINHTTP_PRIVATE_H_ #include #include #define WIN32_NO_STATUS #define _INC_WINDOWS #define COM_NO_WINDOWS_H #define COBJMACROS #define NONAMELESSUNION #include #include #include #include #include #include #include #include #ifdef HAVE_SYS_SOCKET_H # include #endif #ifdef HAVE_NETINET_IN_H # include #endif #ifdef HAVE_NETDB_H # include #endif #if defined(__MINGW32__) || defined (_MSC_VER) # include #else # define closesocket close # define ioctlsocket ioctl #endif #include #include WINE_DEFAULT_DEBUG_CHANNEL(winhttp); static const WCHAR getW[] = {'G','E','T',0}; static const WCHAR postW[] = {'P','O','S','T',0}; static const WCHAR headW[] = {'H','E','A','D',0}; static const WCHAR slashW[] = {'/',0}; static const WCHAR http1_0[] = {'H','T','T','P','/','1','.','0',0}; static const WCHAR http1_1[] = {'H','T','T','P','/','1','.','1',0}; static const WCHAR chunkedW[] = {'c','h','u','n','k','e','d',0}; typedef struct _object_header_t object_header_t; typedef struct { void (*destroy)( object_header_t * ); BOOL (*query_option)( object_header_t *, DWORD, void *, DWORD * ); BOOL (*set_option)( object_header_t *, DWORD, void *, DWORD ); } object_vtbl_t; struct _object_header_t { DWORD type; HINTERNET handle; const object_vtbl_t *vtbl; DWORD flags; DWORD disable_flags; DWORD logon_policy; DWORD redirect_policy; DWORD error; DWORD_PTR context; LONG refs; WINHTTP_STATUS_CALLBACK callback; DWORD notify_mask; struct list entry; struct list children; }; typedef struct { struct list entry; WCHAR *name; struct list cookies; } domain_t; typedef struct { struct list entry; WCHAR *name; WCHAR *value; WCHAR *path; } cookie_t; typedef struct { object_header_t hdr; LPWSTR agent; DWORD access; int resolve_timeout; int connect_timeout; int send_timeout; int recv_timeout; LPWSTR proxy_server; LPWSTR proxy_bypass; LPWSTR proxy_username; LPWSTR proxy_password; struct list cookie_cache; } session_t; typedef struct { object_header_t hdr; session_t *session; LPWSTR hostname; /* final destination of the request */ LPWSTR servername; /* name of the server we directly connect to */ LPWSTR username; LPWSTR password; INTERNET_PORT hostport; INTERNET_PORT serverport; struct sockaddr_storage sockaddr; BOOL resolved; } connect_t; typedef struct { int socket; BOOL secure; /* SSL active on connection? */ CtxtHandle ssl_ctx; SecPkgContext_StreamSizes ssl_sizes; char *ssl_buf; char *extra_buf; size_t extra_len; char *peek_msg; char *peek_msg_mem; size_t peek_len; DWORD security_flags; } netconn_t; typedef struct { LPWSTR field; LPWSTR value; BOOL is_request; /* part of request headers? */ } header_t; enum auth_scheme { SCHEME_INVALID = -1, SCHEME_BASIC, SCHEME_NTLM, SCHEME_PASSPORT, SCHEME_DIGEST, SCHEME_NEGOTIATE }; struct authinfo { enum auth_scheme scheme; CredHandle cred; CtxtHandle ctx; TimeStamp exp; ULONG attr; ULONG max_token; char *data; unsigned int data_len; BOOL finished; /* finished authenticating */ }; typedef struct { object_header_t hdr; connect_t *connect; LPWSTR verb; LPWSTR path; LPWSTR version; LPWSTR raw_headers; void *optional; DWORD optional_len; netconn_t netconn; int resolve_timeout; int connect_timeout; int send_timeout; int recv_timeout; LPWSTR status_text; DWORD content_length; /* total number of bytes to be read (per chunk) */ DWORD content_read; /* bytes read so far */ BOOL read_chunked; /* are we reading in chunked mode? */ DWORD read_pos; /* current read position in read_buf */ DWORD read_size; /* valid data size in read_buf */ char read_buf[4096]; /* buffer for already read but not returned data */ header_t *headers; DWORD num_headers; WCHAR **accept_types; DWORD num_accept_types; struct authinfo *authinfo; struct authinfo *proxy_authinfo; } request_t; typedef struct _task_header_t task_header_t; struct _task_header_t { request_t *request; void (*proc)( task_header_t * ); }; typedef struct { task_header_t hdr; LPWSTR headers; DWORD headers_len; LPVOID optional; DWORD optional_len; DWORD total_len; DWORD_PTR context; } send_request_t; typedef struct { task_header_t hdr; } receive_response_t; typedef struct { task_header_t hdr; LPDWORD available; } query_data_t; typedef struct { task_header_t hdr; LPVOID buffer; DWORD to_read; LPDWORD read; } read_data_t; typedef struct { task_header_t hdr; LPCVOID buffer; DWORD to_write; LPDWORD written; } write_data_t; object_header_t *addref_object( object_header_t * ) DECLSPEC_HIDDEN; object_header_t *grab_object( HINTERNET ) DECLSPEC_HIDDEN; void release_object( object_header_t * ) DECLSPEC_HIDDEN; HINTERNET alloc_handle( object_header_t * ) DECLSPEC_HIDDEN; BOOL free_handle( HINTERNET ) DECLSPEC_HIDDEN; void set_last_error( DWORD ) DECLSPEC_HIDDEN; DWORD get_last_error( void ) DECLSPEC_HIDDEN; void send_callback( object_header_t *, DWORD, LPVOID, DWORD ) DECLSPEC_HIDDEN; void close_connection( request_t * ) DECLSPEC_HIDDEN; BOOL netconn_close( netconn_t * ) DECLSPEC_HIDDEN; BOOL netconn_connect( netconn_t *, const struct sockaddr *, unsigned int, int ) DECLSPEC_HIDDEN; BOOL netconn_connected( netconn_t * ) DECLSPEC_HIDDEN; BOOL netconn_create( netconn_t *, int, int, int ) DECLSPEC_HIDDEN; BOOL netconn_init( netconn_t * ) DECLSPEC_HIDDEN; void netconn_unload( void ) DECLSPEC_HIDDEN; BOOL netconn_query_data_available( netconn_t *, DWORD * ) DECLSPEC_HIDDEN; BOOL netconn_recv( netconn_t *, void *, size_t, int, int * ) DECLSPEC_HIDDEN; BOOL netconn_resolve( WCHAR *, INTERNET_PORT, struct sockaddr *, socklen_t *, int ) DECLSPEC_HIDDEN; BOOL netconn_secure_connect( netconn_t *, WCHAR * ) DECLSPEC_HIDDEN; BOOL netconn_send( netconn_t *, const void *, size_t, int * ) DECLSPEC_HIDDEN; DWORD netconn_set_timeout( netconn_t *, BOOL, int ) DECLSPEC_HIDDEN; const void *netconn_get_certificate( netconn_t * ) DECLSPEC_HIDDEN; int netconn_get_cipher_strength( netconn_t * ) DECLSPEC_HIDDEN; BOOL set_cookies( request_t *, const WCHAR * ) DECLSPEC_HIDDEN; BOOL add_cookie_headers( request_t * ) DECLSPEC_HIDDEN; BOOL add_request_headers( request_t *, LPCWSTR, DWORD, DWORD ) DECLSPEC_HIDDEN; void delete_domain( domain_t * ) DECLSPEC_HIDDEN; BOOL set_server_for_hostname( connect_t *, LPCWSTR, INTERNET_PORT ) DECLSPEC_HIDDEN; void destroy_authinfo( struct authinfo * ) DECLSPEC_HIDDEN; extern HRESULT WinHttpRequest_create( IUnknown *, void ** ) DECLSPEC_HIDDEN; static inline const char *debugstr_variant( const VARIANT *v ) { if (!v) return "(null)"; switch (V_VT(v)) { case VT_EMPTY: return "{VT_EMPTY}"; case VT_NULL: return "{VT_NULL}"; case VT_I4: return wine_dbg_sprintf( "{VT_I4: %d}", V_I4(v) ); case VT_R8: return wine_dbg_sprintf( "{VT_R8: %lf}", V_R8(v) ); case VT_BSTR: return wine_dbg_sprintf( "{VT_BSTR: %s}", debugstr_w(V_BSTR(v)) ); case VT_DISPATCH: return wine_dbg_sprintf( "{VT_DISPATCH: %p}", V_DISPATCH(v) ); case VT_BOOL: return wine_dbg_sprintf( "{VT_BOOL: %x}", V_BOOL(v) ); case VT_UNKNOWN: return wine_dbg_sprintf( "{VT_UNKNOWN: %p}", V_UNKNOWN(v) ); case VT_UINT: return wine_dbg_sprintf( "{VT_UINT: %u}", V_UINT(v) ); case VT_BSTR|VT_BYREF: return wine_dbg_sprintf( "{VT_BSTR|VT_BYREF: ptr %p, data %s}", V_BSTRREF(v), V_BSTRREF(v) ? debugstr_w( *V_BSTRREF(v) ) : NULL ); default: return wine_dbg_sprintf( "{vt %d}", V_VT(v) ); } } static inline void *heap_alloc( SIZE_T size ) { return HeapAlloc( GetProcessHeap(), 0, size ); } static inline void *heap_alloc_zero( SIZE_T size ) { return HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, size ); } static inline void *heap_realloc( LPVOID mem, SIZE_T size ) { return HeapReAlloc( GetProcessHeap(), 0, mem, size ); } static inline void *heap_realloc_zero( LPVOID mem, SIZE_T size ) { return HeapReAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, mem, size ); } static inline BOOL heap_free( LPVOID mem ) { return HeapFree( GetProcessHeap(), 0, mem ); } static inline WCHAR *strdupW( const WCHAR *src ) { WCHAR *dst; if (!src) return NULL; dst = heap_alloc( (strlenW( src ) + 1) * sizeof(WCHAR) ); if (dst) strcpyW( dst, src ); return dst; } static inline WCHAR *strdupAW( const char *src ) { WCHAR *dst = NULL; if (src) { DWORD len = MultiByteToWideChar( CP_ACP, 0, src, -1, NULL, 0 ); if ((dst = heap_alloc( len * sizeof(WCHAR) ))) MultiByteToWideChar( CP_ACP, 0, src, -1, dst, len ); } return dst; } static inline char *strdupWA( const WCHAR *src ) { char *dst = NULL; if (src) { int len = WideCharToMultiByte( CP_ACP, 0, src, -1, NULL, 0, NULL, NULL ); if ((dst = heap_alloc( len ))) WideCharToMultiByte( CP_ACP, 0, src, -1, dst, len, NULL, NULL ); } return dst; } #endif /* _WINE_WINHTTP_PRIVATE_H_ */