256 lines
6.2 KiB
C
256 lines
6.2 KiB
C
/***************************************************************************
|
|
* _ _ ____ _
|
|
* Project ___| | | | _ \| |
|
|
* / __| | | | |_) | |
|
|
* | (__| |_| | _ <| |___
|
|
* \___|\___/|_| \_\_____|
|
|
*
|
|
* Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
|
|
*
|
|
* This software is licensed as described in the file COPYING, which
|
|
* you should have received as part of this distribution. The terms
|
|
* are also available at https://curl.se/docs/copyright.html.
|
|
*
|
|
* You may opt to use, copy, modify, merge, publish, distribute and/or sell
|
|
* copies of the Software, and permit persons to whom the Software is
|
|
* furnished to do so, under the terms of the COPYING file.
|
|
*
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
* KIND, either express or implied.
|
|
*
|
|
* SPDX-License-Identifier: curl
|
|
*
|
|
***************************************************************************/
|
|
#include "first.h"
|
|
|
|
#include "testtrace.h"
|
|
#include "memdebug.h"
|
|
|
|
#ifndef CURL_DISABLE_WEBSOCKETS
|
|
|
|
static const char *descr_flags(int flags)
|
|
{
|
|
if(flags & CURLWS_TEXT)
|
|
return flags & CURLWS_CONT ? "txt ---" : "txt fin";
|
|
if(flags & CURLWS_BINARY)
|
|
return flags & CURLWS_CONT ? "bin ---" : "bin fin";
|
|
if(flags & CURLWS_PING)
|
|
return "ping";
|
|
if(flags & CURLWS_PONG)
|
|
return "pong";
|
|
if(flags & CURLWS_CLOSE)
|
|
return "close";
|
|
assert(false);
|
|
return "";
|
|
}
|
|
|
|
static CURLcode send_header(CURL *curl, int flags, size_t size)
|
|
{
|
|
CURLcode res = CURLE_OK;
|
|
size_t nsent;
|
|
|
|
retry:
|
|
res = curl_ws_send(curl, NULL, 0, &nsent, (curl_off_t)size,
|
|
flags | CURLWS_OFFSET);
|
|
if(res == CURLE_AGAIN) {
|
|
assert(nsent == 0);
|
|
goto retry;
|
|
}
|
|
if(res) {
|
|
curl_mfprintf(stderr, "%s:%d curl_ws_send() failed with code %d (%s)\n",
|
|
__FILE__, __LINE__, res, curl_easy_strerror(res));
|
|
assert(nsent == 0);
|
|
return res;
|
|
}
|
|
|
|
assert(nsent == 0);
|
|
|
|
return CURLE_OK;
|
|
}
|
|
|
|
static CURLcode recv_header(CURL *curl, int *flags, curl_off_t *offset,
|
|
curl_off_t *bytesleft)
|
|
{
|
|
CURLcode res = CURLE_OK;
|
|
size_t nread;
|
|
const struct curl_ws_frame *meta;
|
|
|
|
*flags = 0;
|
|
*offset = 0;
|
|
*bytesleft = 0;
|
|
|
|
retry:
|
|
res = curl_ws_recv(curl, NULL, 0, &nread, &meta);
|
|
if(res == CURLE_AGAIN) {
|
|
assert(nread == 0);
|
|
goto retry;
|
|
}
|
|
if(res) {
|
|
curl_mfprintf(stderr, "%s:%d curl_ws_recv() failed with code %d (%s)\n",
|
|
__FILE__, __LINE__, res, curl_easy_strerror(res));
|
|
assert(nread == 0);
|
|
return res;
|
|
}
|
|
|
|
assert(nread == 0);
|
|
assert(meta != NULL);
|
|
assert(meta->flags);
|
|
assert(meta->offset == 0);
|
|
|
|
*flags = meta->flags;
|
|
*offset = meta->offset;
|
|
*bytesleft = meta->bytesleft;
|
|
|
|
curl_mfprintf(stdout, "%s [%" FMT_OFF_T "]", descr_flags(meta->flags),
|
|
meta->bytesleft);
|
|
|
|
if(meta->bytesleft > 0)
|
|
curl_mfprintf(stdout, " ");
|
|
|
|
res = send_header(curl, meta->flags, (size_t)meta->bytesleft);
|
|
if(res)
|
|
return res;
|
|
|
|
return CURLE_OK;
|
|
}
|
|
|
|
static CURLcode send_chunk(CURL *curl, int flags, const char *buffer,
|
|
size_t size, size_t *offset)
|
|
{
|
|
CURLcode res = CURLE_OK;
|
|
size_t nsent;
|
|
|
|
retry:
|
|
res = curl_ws_send(curl, buffer + *offset, size - *offset, &nsent, 0,
|
|
flags);
|
|
if(res == CURLE_AGAIN) {
|
|
assert(nsent == 0);
|
|
goto retry;
|
|
}
|
|
if(res) {
|
|
curl_mfprintf(stderr, "%s:%d curl_ws_send() failed with code %d (%s)\n",
|
|
__FILE__, __LINE__, res, curl_easy_strerror(res));
|
|
assert(nsent == 0);
|
|
return res;
|
|
}
|
|
|
|
assert(nsent <= size - *offset);
|
|
|
|
*offset += nsent;
|
|
|
|
return CURLE_OK;
|
|
}
|
|
|
|
static CURLcode recv_chunk(CURL *curl, int flags, curl_off_t *offset,
|
|
curl_off_t *bytesleft)
|
|
{
|
|
CURLcode res = CURLE_OK;
|
|
char buffer[256];
|
|
size_t nread;
|
|
const struct curl_ws_frame *meta;
|
|
size_t sendoffset = 0;
|
|
|
|
retry:
|
|
res = curl_ws_recv(curl, buffer, sizeof(buffer), &nread, &meta);
|
|
if(res == CURLE_AGAIN) {
|
|
assert(nread == 0);
|
|
goto retry;
|
|
}
|
|
if(res) {
|
|
curl_mfprintf(stderr, "%s:%d curl_ws_recv() failed with code %d (%s)\n",
|
|
__FILE__, __LINE__, res, curl_easy_strerror(res));
|
|
assert(nread == 0);
|
|
return res;
|
|
}
|
|
|
|
assert(nread <= sizeof(buffer));
|
|
assert(meta != NULL);
|
|
assert(meta->flags == flags);
|
|
assert(meta->offset == *offset);
|
|
assert(meta->bytesleft == (*bytesleft - (curl_off_t)nread));
|
|
|
|
*offset += nread;
|
|
*bytesleft -= nread;
|
|
|
|
fwrite(buffer, 1, nread, stdout);
|
|
|
|
while(sendoffset < nread) {
|
|
res = send_chunk(curl, flags, buffer, nread, &sendoffset);
|
|
if(res)
|
|
return res;
|
|
}
|
|
|
|
return CURLE_OK;
|
|
}
|
|
|
|
static CURLcode recv_frame(CURL *curl, bool *stop)
|
|
{
|
|
CURLcode res = CURLE_OK;
|
|
int flags = 0;
|
|
curl_off_t offset = 0;
|
|
curl_off_t bytesleft = 0;
|
|
|
|
res = recv_header(curl, &flags, &offset, &bytesleft);
|
|
if(res)
|
|
return res;
|
|
|
|
while(bytesleft > 0) {
|
|
res = recv_chunk(curl, flags, &offset, &bytesleft);
|
|
if(res)
|
|
return res;
|
|
}
|
|
|
|
if(flags & CURLWS_CLOSE)
|
|
*stop = true;
|
|
|
|
curl_mfprintf(stdout, "\n");
|
|
|
|
return res;
|
|
}
|
|
#endif
|
|
|
|
static CURLcode test_lib2700(const char *URL)
|
|
{
|
|
#ifndef CURL_DISABLE_WEBSOCKETS
|
|
CURLcode res = CURLE_OK;
|
|
bool stop = false;
|
|
CURL *curl;
|
|
|
|
global_init(CURL_GLOBAL_ALL);
|
|
curl_global_trace("ws");
|
|
easy_init(curl);
|
|
|
|
easy_setopt(curl, CURLOPT_URL, URL);
|
|
easy_setopt(curl, CURLOPT_USERAGENT, "client/test2700");
|
|
debug_config.nohex = TRUE;
|
|
debug_config.tracetime = TRUE;
|
|
easy_setopt(curl, CURLOPT_DEBUGDATA, &debug_config);
|
|
easy_setopt(curl, CURLOPT_DEBUGFUNCTION, libtest_debug_cb);
|
|
easy_setopt(curl, CURLOPT_VERBOSE, 1L);
|
|
easy_setopt(curl, CURLOPT_CONNECT_ONLY, 2L);
|
|
if(testnum != 2708)
|
|
easy_setopt(curl, CURLOPT_WS_OPTIONS, CURLWS_NOAUTOPONG);
|
|
|
|
res = curl_easy_perform(curl);
|
|
if(res) {
|
|
curl_mfprintf(stderr,
|
|
"%s:%d curl_easy_perform() failed with code %d (%s)\n",
|
|
__FILE__, __LINE__, res, curl_easy_strerror(res));
|
|
goto test_cleanup;
|
|
}
|
|
|
|
while(!stop) {
|
|
res = recv_frame(curl, &stop);
|
|
if(res)
|
|
goto test_cleanup;
|
|
}
|
|
|
|
test_cleanup:
|
|
curl_easy_cleanup(curl);
|
|
curl_global_cleanup();
|
|
return res;
|
|
#else
|
|
NO_SUPPORT_BUILT_IN
|
|
#endif
|
|
}
|