| /* |
| * Copyright 2016 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 |
| */ |
| |
| #include <stdio.h> |
| #include "windows.h" |
| #include "webservices.h" |
| #include "wine/test.h" |
| |
| static void test_WsCreateChannel(void) |
| { |
| HRESULT hr; |
| WS_CHANNEL *channel; |
| WS_CHANNEL_STATE state; |
| WS_CHANNEL_PROPERTY prop; |
| WS_ENCODING encoding; |
| WS_ENVELOPE_VERSION env_version; |
| WS_ADDRESSING_VERSION addr_version; |
| ULONG size; |
| |
| hr = WsCreateChannel( WS_CHANNEL_TYPE_REQUEST, WS_HTTP_CHANNEL_BINDING, NULL, 0, NULL, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| channel = NULL; |
| hr = WsCreateChannel( WS_CHANNEL_TYPE_REQUEST, WS_HTTP_CHANNEL_BINDING, NULL, 0, NULL, &channel, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( channel != NULL, "channel not set\n" ); |
| |
| size = 0xdeadbeef; |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_MAX_BUFFERED_MESSAGE_SIZE, &size, sizeof(size), NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( size == 65536, "got %u\n", size ); |
| |
| encoding = 0xdeadbeef; |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_ENCODING, &encoding, sizeof(encoding), NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( encoding == WS_ENCODING_XML_UTF8, "got %u\n", encoding ); |
| |
| env_version = 0xdeadbeef; |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_ENVELOPE_VERSION, &env_version, sizeof(env_version), |
| NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( env_version == WS_ENVELOPE_VERSION_SOAP_1_2, "got %u\n", env_version ); |
| |
| addr_version = 0xdeadbeef; |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_ADDRESSING_VERSION, &addr_version, sizeof(addr_version), |
| NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( addr_version == WS_ADDRESSING_VERSION_1_0, "got %u\n", addr_version ); |
| |
| /* read-only property */ |
| state = 0xdeadbeef; |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_STATE, &state, sizeof(state), NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( state == WS_CHANNEL_STATE_CREATED, "got %u\n", state ); |
| |
| state = WS_CHANNEL_STATE_CREATED; |
| hr = WsSetChannelProperty( channel, WS_CHANNEL_PROPERTY_STATE, &state, sizeof(state), NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| encoding = WS_ENCODING_XML_UTF8; |
| hr = WsSetChannelProperty( channel, WS_CHANNEL_PROPERTY_ENCODING, &encoding, sizeof(encoding), NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| WsFreeChannel( channel ); |
| |
| encoding = WS_ENCODING_XML_UTF16LE; |
| prop.id = WS_CHANNEL_PROPERTY_ENCODING; |
| prop.value = &encoding; |
| prop.valueSize = sizeof(encoding); |
| hr = WsCreateChannel( WS_CHANNEL_TYPE_REQUEST, WS_HTTP_CHANNEL_BINDING, &prop, 1, NULL, &channel, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| encoding = 0xdeadbeef; |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_ENCODING, &encoding, sizeof(encoding), NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( encoding == WS_ENCODING_XML_UTF16LE, "got %u\n", encoding ); |
| WsFreeChannel( channel ); |
| |
| hr = WsCreateChannel( WS_CHANNEL_TYPE_DUPLEX_SESSION, WS_TCP_CHANNEL_BINDING, NULL, 0, NULL, &channel, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| encoding = 0xdeadbeef; |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_ENCODING, &encoding, sizeof(encoding), NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( encoding == WS_ENCODING_XML_BINARY_SESSION_1, "got %u\n", encoding ); |
| |
| env_version = 0xdeadbeef; |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_ENVELOPE_VERSION, &env_version, sizeof(env_version), |
| NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( env_version == WS_ENVELOPE_VERSION_SOAP_1_2, "got %u\n", env_version ); |
| |
| addr_version = 0xdeadbeef; |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_ADDRESSING_VERSION, &addr_version, sizeof(addr_version), |
| NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( addr_version == WS_ADDRESSING_VERSION_1_0, "got %u\n", addr_version ); |
| WsFreeChannel( channel ); |
| } |
| |
| static void test_WsOpenChannel(void) |
| { |
| WCHAR url[] = {'h','t','t','p',':','/','/','l','o','c','a','l','h','o','s','t'}; |
| HRESULT hr; |
| WS_CHANNEL *channel; |
| WS_ENDPOINT_ADDRESS addr; |
| |
| hr = WsCreateChannel( WS_CHANNEL_TYPE_REQUEST, WS_HTTP_CHANNEL_BINDING, NULL, 0, NULL, &channel, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsCloseChannel( channel, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| WsFreeChannel( channel ); |
| |
| hr = WsCreateChannel( WS_CHANNEL_TYPE_REQUEST, WS_HTTP_CHANNEL_BINDING, NULL, 0, NULL, &channel, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsOpenChannel( channel, NULL, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| memset( &addr, 0, sizeof(addr) ); |
| addr.url.length = sizeof(url)/sizeof(url[0]); |
| addr.url.chars = url; |
| hr = WsOpenChannel( NULL, &addr, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsOpenChannel( channel, &addr, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsOpenChannel( channel, &addr, NULL, NULL ); |
| ok( hr == WS_E_INVALID_OPERATION, "got %08x\n", hr ); |
| |
| hr = WsCloseChannel( channel, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsCloseChannel( channel, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsCloseChannel( NULL, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| WsFreeChannel( channel ); |
| } |
| |
| static void test_WsResetChannel(void) |
| { |
| WCHAR url[] = {'h','t','t','p',':','/','/','l','o','c','a','l','h','o','s','t'}; |
| HRESULT hr; |
| WS_CHANNEL *channel; |
| WS_CHANNEL_STATE state; |
| WS_CHANNEL_TYPE type; |
| WS_ENDPOINT_ADDRESS addr; |
| ULONG size, timeout; |
| |
| hr = WsCreateChannel( WS_CHANNEL_TYPE_REQUEST, WS_HTTP_CHANNEL_BINDING, NULL, 0, NULL, &channel, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsResetChannel( channel, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| timeout = 5000; |
| size = sizeof(timeout); |
| hr = WsSetChannelProperty( channel, WS_CHANNEL_PROPERTY_RESOLVE_TIMEOUT, &timeout, size, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| memset( &addr, 0, sizeof(addr) ); |
| addr.url.length = sizeof(url)/sizeof(url[0]); |
| addr.url.chars = url; |
| hr = WsOpenChannel( channel, &addr, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsResetChannel( channel, NULL ); |
| ok( hr == WS_E_INVALID_OPERATION, "got %08x\n", hr ); |
| |
| hr = WsCloseChannel( channel, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsResetChannel( channel, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| state = 0xdeadbeef; |
| size = sizeof(state); |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_STATE, &state, size, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( state == WS_CHANNEL_STATE_CREATED, "got %u\n", state ); |
| |
| type = 0xdeadbeef; |
| size = sizeof(type); |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_CHANNEL_TYPE, &type, size, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( type == WS_CHANNEL_TYPE_REQUEST, "got %u\n", type ); |
| |
| timeout = 0xdeadbeef; |
| size = sizeof(timeout); |
| hr = WsGetChannelProperty( channel, WS_CHANNEL_PROPERTY_RESOLVE_TIMEOUT, &timeout, size, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( timeout == 5000, "got %u\n", timeout ); |
| |
| WsFreeChannel( channel ); |
| } |
| |
| struct listener_info |
| { |
| int port; |
| HANDLE wait; |
| WS_CHANNEL_BINDING binding; |
| WS_CHANNEL_TYPE type; |
| void (*test_func)( WS_CHANNEL * ); |
| }; |
| |
| static void client_message_read_write( WS_CHANNEL *channel ) |
| { |
| WS_MESSAGE *msg; |
| HRESULT hr; |
| |
| hr = WsCreateMessageForChannel( channel, NULL, 0, &msg, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsReadMessageStart( NULL, NULL, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsReadMessageStart( channel, NULL, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsReadMessageStart( NULL, msg, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsReadMessageStart( channel, msg, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsReadMessageEnd( NULL, NULL, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsReadMessageEnd( channel, NULL, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsReadMessageEnd( NULL, msg, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsReadMessageEnd( channel, msg, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| WsFreeMessage( msg ); |
| } |
| |
| static void test_message_read_write( const struct listener_info *info ) |
| { |
| static const WCHAR fmt[] = |
| {'s','o','a','p','.','u','d','p',':','/','/','l','o','c','a','l','h','o','s','t',':','%','u',0}; |
| WS_ENDPOINT_ADDRESS addr; |
| WCHAR buf[64]; |
| WS_CHANNEL *channel; |
| WS_MESSAGE *msg; |
| HRESULT hr; |
| DWORD err; |
| |
| hr = WsCreateChannel( info->type, info->binding, NULL, 0, NULL, &channel, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| memset( &addr, 0, sizeof(addr) ); |
| addr.url.length = wsprintfW( buf, fmt, info->port ); |
| addr.url.chars = buf; |
| hr = WsOpenChannel( channel, &addr, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsCreateMessageForChannel( channel, NULL, 0, &msg, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsInitializeMessage( msg, WS_REQUEST_MESSAGE, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsWriteMessageStart( NULL, NULL, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsWriteMessageStart( channel, NULL, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsWriteMessageStart( NULL, msg, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsWriteMessageStart( channel, msg, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsWriteMessageEnd( NULL, NULL, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsWriteMessageEnd( channel, NULL, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsWriteMessageEnd( NULL, msg, NULL, NULL ); |
| ok( hr == E_INVALIDARG, "got %08x\n", hr ); |
| |
| hr = WsWriteMessageEnd( channel, msg, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| err = WaitForSingleObject( info->wait, 3000 ); |
| ok( err == WAIT_OBJECT_0, "wait failed %u\n", err ); |
| |
| hr = WsCloseChannel( channel, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| WsFreeMessage( msg ); |
| WsFreeChannel( channel ); |
| } |
| |
| struct async_test |
| { |
| ULONG call_count; |
| HANDLE wait; |
| }; |
| |
| static void CALLBACK callback_duplex_session( HRESULT hr, WS_CALLBACK_MODEL model, void *state ) |
| { |
| struct async_test *test = state; |
| |
| ok( hr == S_OK, "got %08x\n", hr ); |
| ok( model == WS_LONG_CALLBACK, "got %u\n", model ); |
| test->call_count++; |
| SetEvent( test->wait ); |
| } |
| |
| static void client_duplex_session( WS_CHANNEL *channel ) |
| { |
| WS_XML_STRING action = {6, (BYTE *)"action"}, localname = {9, (BYTE *)"localname"}, ns = {2, (BYTE *)"ns"}; |
| WS_ELEMENT_DESCRIPTION desc_body; |
| const WS_MESSAGE_DESCRIPTION *desc[1]; |
| WS_MESSAGE_DESCRIPTION desc_msg; |
| struct async_test test; |
| WS_ASYNC_CONTEXT ctx; |
| WS_MESSAGE *msg, *msg2; |
| INT32 val = 0; |
| HRESULT hr; |
| |
| hr = WsCreateMessageForChannel( channel, NULL, 0, &msg, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| desc_body.elementLocalName = &localname; |
| desc_body.elementNs = &ns; |
| desc_body.type = WS_INT32_TYPE; |
| desc_body.typeDescription = NULL; |
| desc_msg.action = &action; |
| desc_msg.bodyElementDescription = &desc_body; |
| desc[0] = &desc_msg; |
| |
| test.call_count = 0; |
| test.wait = CreateEventW( NULL, FALSE, FALSE, NULL ); |
| ctx.callback = callback_duplex_session; |
| ctx.callbackState = &test; |
| |
| hr = WsReceiveMessage( channel, msg, desc, 1, WS_RECEIVE_OPTIONAL_MESSAGE, WS_READ_REQUIRED_VALUE, |
| NULL, &val, sizeof(val), NULL, &ctx, NULL ); |
| ok( hr == WS_S_ASYNC || hr == S_OK, "got %08x\n", hr ); |
| if (hr == WS_S_ASYNC) |
| { |
| WaitForSingleObject( test.wait, INFINITE ); |
| ok( test.call_count == 1, "got %u\n", test.call_count ); |
| } |
| else ok( !test.call_count, "got %u\n", test.call_count ); |
| |
| hr = WsCreateMessageForChannel( channel, NULL, 0, &msg2, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| test.call_count = 0; |
| hr = WsReceiveMessage( channel, msg2, desc, 1, WS_RECEIVE_OPTIONAL_MESSAGE, WS_READ_REQUIRED_VALUE, |
| NULL, &val, sizeof(val), NULL, &ctx, NULL ); |
| ok( hr == WS_S_ASYNC || hr == S_OK, "got %08x\n", hr ); |
| if (hr == WS_S_ASYNC) |
| { |
| WaitForSingleObject( test.wait, INFINITE ); |
| ok( test.call_count == 1, "got %u\n", test.call_count ); |
| } |
| else ok( !test.call_count, "got %u\n", test.call_count ); |
| |
| CloseHandle( test.wait ); |
| WsFreeMessage( msg ); |
| WsFreeMessage( msg2 ); |
| } |
| |
| static void test_duplex_session( const struct listener_info *info ) |
| { |
| static const WCHAR fmt[] = |
| {'n','e','t','.','t','c','p',':','/','/','l','o','c','a','l','h','o','s','t',':','%','u',0}; |
| WS_XML_STRING action = {6, (BYTE *)"action"}, localname = {9, (BYTE *)"localname"}, ns = {2, (BYTE *)"ns"}; |
| WS_ELEMENT_DESCRIPTION desc_body; |
| WS_MESSAGE_DESCRIPTION desc; |
| WS_ENDPOINT_ADDRESS addr; |
| WCHAR buf[64]; |
| WS_CHANNEL *channel; |
| WS_MESSAGE *msg, *msg2; |
| INT32 val = -1; |
| HRESULT hr; |
| DWORD err; |
| |
| hr = WsCreateChannel( info->type, info->binding, NULL, 0, NULL, &channel, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| memset( &addr, 0, sizeof(addr) ); |
| addr.url.length = wsprintfW( buf, fmt, info->port ); |
| addr.url.chars = buf; |
| hr = WsOpenChannel( channel, &addr, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsCreateMessageForChannel( channel, NULL, 0, &msg, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| desc_body.elementLocalName = &localname; |
| desc_body.elementNs = &ns; |
| desc_body.type = WS_INT32_TYPE; |
| desc_body.typeDescription = NULL; |
| desc.action = &action; |
| desc.bodyElementDescription = &desc_body; |
| |
| hr = WsSendMessage( channel, msg, &desc, WS_WRITE_REQUIRED_VALUE, &val, sizeof(val), NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsCreateMessageForChannel( channel, NULL, 0, &msg2, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsSendMessage( channel, msg2, &desc, WS_WRITE_REQUIRED_VALUE, &val, sizeof(val), NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| err = WaitForSingleObject( info->wait, 3000 ); |
| ok( err == WAIT_OBJECT_0, "wait failed %u\n", err ); |
| |
| hr = WsCloseChannel( channel, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| WsFreeMessage( msg ); |
| WsFreeMessage( msg2 ); |
| WsFreeChannel( channel ); |
| } |
| |
| static DWORD CALLBACK listener_proc( void *arg ) |
| { |
| static const WCHAR fmt_tcp[] = |
| {'n','e','t','.','t','c','p',':','/','/','l','o','c','a','l','h','o','s','t',':','%','u',0}; |
| static const WCHAR fmt_udp[] = |
| {'s','o','a','p','.','u','d','p',':','/','/','l','o','c','a','l','h','o','s','t',':','%','u',0}; |
| struct listener_info *info = arg; |
| const WCHAR *fmt = (info->binding == WS_TCP_CHANNEL_BINDING) ? fmt_tcp : fmt_udp; |
| WS_LISTENER *listener; |
| WS_CHANNEL *channel; |
| WCHAR buf[64]; |
| WS_STRING url; |
| HRESULT hr; |
| |
| hr = WsCreateListener( info->type, info->binding, NULL, 0, NULL, &listener, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| url.length = wsprintfW( buf, fmt, info->port ); |
| url.chars = buf; |
| hr = WsOpenListener( listener, &url, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| hr = WsCreateChannelForListener( listener, NULL, 0, &channel, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| SetEvent( info->wait ); |
| |
| hr = WsAcceptChannel( listener, channel, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| |
| info->test_func( channel ); |
| |
| SetEvent( info->wait ); |
| |
| hr = WsCloseChannel( channel, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| WsFreeChannel( channel ); |
| |
| hr = WsCloseListener( listener, NULL, NULL ); |
| ok( hr == S_OK, "got %08x\n", hr ); |
| WsFreeListener( listener ); |
| |
| return 0; |
| } |
| |
| static HANDLE start_listener( struct listener_info *info ) |
| { |
| DWORD err; |
| HANDLE thread = CreateThread( NULL, 0, listener_proc, info, 0, NULL ); |
| ok( thread != NULL, "failed to create listener thread %u\n", GetLastError() ); |
| |
| err = WaitForSingleObject( info->wait, 3000 ); |
| ok( err == WAIT_OBJECT_0, "failed to start listener %u\n", err ); |
| return thread; |
| } |
| |
| START_TEST(channel) |
| { |
| struct listener_info info; |
| HANDLE thread; |
| |
| test_WsCreateChannel(); |
| test_WsOpenChannel(); |
| test_WsResetChannel(); |
| |
| info.port = 7533; |
| info.wait = CreateEventW( NULL, 0, 0, NULL ); |
| info.type = WS_CHANNEL_TYPE_DUPLEX; |
| info.binding = WS_UDP_CHANNEL_BINDING; |
| info.test_func = client_message_read_write; |
| |
| thread = start_listener( &info ); |
| test_message_read_write( &info ); |
| WaitForSingleObject( thread, 3000 ); |
| |
| info.type = WS_CHANNEL_TYPE_DUPLEX_SESSION; |
| info.binding = WS_TCP_CHANNEL_BINDING; |
| info.test_func = client_duplex_session; |
| |
| thread = start_listener( &info ); |
| test_duplex_session( &info ); |
| WaitForSingleObject( thread, 3000 ); |
| } |