Commit f3346a78 authored by Hans Leidekker's avatar Hans Leidekker Committed by Alexandre Julliard

winhttp: Support asynchronous requests.

parent a4d0abb2
...@@ -34,6 +34,9 @@ enum api ...@@ -34,6 +34,9 @@ enum api
winhttp_open_request, winhttp_open_request,
winhttp_send_request, winhttp_send_request,
winhttp_receive_response, winhttp_receive_response,
winhttp_query_data,
winhttp_read_data,
winhttp_write_data,
winhttp_close_handle winhttp_close_handle
}; };
...@@ -70,12 +73,15 @@ static void CALLBACK check_notification( HINTERNET handle, DWORD_PTR context, DW ...@@ -70,12 +73,15 @@ static void CALLBACK check_notification( HINTERNET handle, DWORD_PTR context, DW
ok(info->test[i].status == status, "expected status 0x%08x got 0x%08x\n", info->test[i].status, status); ok(info->test[i].status == status, "expected status 0x%08x got 0x%08x\n", info->test[i].status, status);
ok(info->test[i].function == info->function, "expected function %u got %u\n", info->test[i].function, info->function); ok(info->test[i].function == info->function, "expected function %u got %u\n", info->test[i].function, info->function);
} }
else todo_wine else
{ {
ok(info->test[i].status == status, "expected status 0x%08x got 0x%08x\n", info->test[i].status, status); todo_wine ok(info->test[i].status == status, "expected status 0x%08x got 0x%08x\n", info->test[i].status, status);
ok(info->test[i].function == info->function, "expected function %u got %u\n", info->test[i].function, info->function); if (info->test[i].status == status)
{
todo_wine ok(info->test[i].function == info->function, "expected function %u got %u\n", info->test[i].function, info->function);
} }
info->index++; }
if (info->test[i].status == status) info->index++;
if (status & WINHTTP_CALLBACK_FLAG_ALL_COMPLETIONS) SetEvent( info->wait ); if (status & WINHTTP_CALLBACK_FLAG_ALL_COMPLETIONS) SetEvent( info->wait );
} }
...@@ -146,7 +152,7 @@ static void test_connection_cache( void ) ...@@ -146,7 +152,7 @@ static void test_connection_cache( void )
WinHttpCloseHandle( con ); WinHttpCloseHandle( con );
WinHttpCloseHandle( ses ); WinHttpCloseHandle( ses );
Sleep(2000); /* make sure connection is evicted from cache */ Sleep(1500); /* make sure connection is evicted from cache */
info.index = 0; info.index = 0;
...@@ -287,6 +293,10 @@ static const struct notification async_test[] = ...@@ -287,6 +293,10 @@ static const struct notification async_test[] =
{ winhttp_receive_response, WINHTTP_CALLBACK_STATUS_RECEIVING_RESPONSE, 0 }, { winhttp_receive_response, WINHTTP_CALLBACK_STATUS_RECEIVING_RESPONSE, 0 },
{ winhttp_receive_response, WINHTTP_CALLBACK_STATUS_RESPONSE_RECEIVED, 0 }, { winhttp_receive_response, WINHTTP_CALLBACK_STATUS_RESPONSE_RECEIVED, 0 },
{ winhttp_receive_response, WINHTTP_CALLBACK_STATUS_HEADERS_AVAILABLE, 0 }, { winhttp_receive_response, WINHTTP_CALLBACK_STATUS_HEADERS_AVAILABLE, 0 },
{ winhttp_query_data, WINHTTP_CALLBACK_STATUS_DATA_AVAILABLE, 0 },
{ winhttp_read_data, WINHTTP_CALLBACK_STATUS_RECEIVING_RESPONSE, 1 },
{ winhttp_read_data, WINHTTP_CALLBACK_STATUS_RESPONSE_RECEIVED, 1 },
{ winhttp_read_data, WINHTTP_CALLBACK_STATUS_READ_COMPLETE, 1 },
{ winhttp_close_handle, WINHTTP_CALLBACK_STATUS_CLOSING_CONNECTION, 0 }, { winhttp_close_handle, WINHTTP_CALLBACK_STATUS_CLOSING_CONNECTION, 0 },
{ winhttp_close_handle, WINHTTP_CALLBACK_STATUS_CONNECTION_CLOSED, 0 }, { winhttp_close_handle, WINHTTP_CALLBACK_STATUS_CONNECTION_CLOSED, 0 },
{ winhttp_close_handle, WINHTTP_CALLBACK_STATUS_HANDLE_CLOSING, 0 }, { winhttp_close_handle, WINHTTP_CALLBACK_STATUS_HANDLE_CLOSING, 0 },
...@@ -302,6 +312,7 @@ static void test_async( void ) ...@@ -302,6 +312,7 @@ static void test_async( void )
DWORD size, status; DWORD size, status;
BOOL ret; BOOL ret;
struct info info, *context = &info; struct info info, *context = &info;
char buffer[1024];
info.test = async_test; info.test = async_test;
info.count = sizeof(async_test) / sizeof(async_test[0]); info.count = sizeof(async_test) / sizeof(async_test[0]);
...@@ -341,6 +352,18 @@ static void test_async( void ) ...@@ -341,6 +352,18 @@ static void test_async( void )
ok(ret, "failed unexpectedly %u\n", GetLastError()); ok(ret, "failed unexpectedly %u\n", GetLastError());
ok(status == 200, "request failed unexpectedly %u\n", status); ok(status == 200, "request failed unexpectedly %u\n", status);
info.function = winhttp_query_data;
ret = WinHttpQueryDataAvailable( req, NULL );
ok(ret, "failed to query data available %u\n", GetLastError());
WaitForSingleObject( info.wait, INFINITE );
info.function = winhttp_read_data;
ret = WinHttpReadData( req, buffer, sizeof(buffer), NULL );
ok(ret, "failed to query data available %u\n", GetLastError());
WaitForSingleObject( info.wait, INFINITE );
info.function = winhttp_close_handle; info.function = winhttp_close_handle;
WinHttpCloseHandle( req ); WinHttpCloseHandle( req );
WinHttpCloseHandle( con ); WinHttpCloseHandle( con );
...@@ -352,5 +375,6 @@ START_TEST (notification) ...@@ -352,5 +375,6 @@ START_TEST (notification)
{ {
test_connection_cache(); test_connection_cache();
test_redirect(); test_redirect();
Sleep(1500); /* make sure previous connection is evicted from cache */
test_async(); test_async();
} }
...@@ -120,6 +120,52 @@ typedef struct ...@@ -120,6 +120,52 @@ typedef struct
DWORD num_headers; DWORD num_headers;
} request_t; } 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 * ); object_header_t *addref_object( object_header_t * );
object_header_t *grab_object( HINTERNET ); object_header_t *grab_object( HINTERNET );
void release_object( object_header_t * ); void release_object( object_header_t * );
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment