mirror of
https://github.com/curl/curl.git
synced 2026-04-14 13:01:42 +08:00
libtests and clients were built the same way after recent overhauls. libtests are used by runtests, clients by pytests. Merge clients into libtests, aligning their entry function signature, dropping common utility functions, and simplifying the build. Note: After this patch `CURLDEBUG` applies to cli tests, when enabled. Also: - lib552: drop local copy-paste debug callback in favor of testtrace. - lib552: drop local copy-paste dump function in favor of testtrace. - clients: use `long` for HTTP version, drop casts. - clients: replace local dump function in favor of testrace clone. - sync cli test entry function prototype with libtests'. - h2_serverpush: replace local trace callback with testtrace. - de-duplicate 3 websocket close, ping, ping, functions. Kept the pong iteration from `ws_pingpong`. Note: the pong clone in `lib2304` was returning an error when `curl_ws_recv()` returned non-zero and the payload matched the expected one anyway. After this patch, this case returns success, as it does in `ws_pingpong`. `lib2304` keeps passing, but I'm not sure if the previous behavior was intentional. - display full value in websocket close, ping, pong, drop casts. Closes #18079
293 lines
7.1 KiB
C
293 lines
7.1 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"
|
|
|
|
#ifdef HAVE_LOCALE_H
|
|
#include <locale.h> /* for setlocale() */
|
|
#endif
|
|
|
|
#include "memdebug.h"
|
|
|
|
int select_wrapper(int nfds, fd_set *rd, fd_set *wr, fd_set *exc,
|
|
struct timeval *tv)
|
|
{
|
|
if(nfds < 0) {
|
|
SET_SOCKERRNO(SOCKEINVAL);
|
|
return -1;
|
|
}
|
|
#ifdef USE_WINSOCK
|
|
/*
|
|
* Winsock select() requires that at least one of the three fd_set
|
|
* pointers is not NULL and points to a non-empty fdset. IOW Winsock
|
|
* select() can not be used to sleep without a single fd_set.
|
|
*/
|
|
if(!nfds) {
|
|
Sleep((DWORD)curlx_tvtoms(tv));
|
|
return 0;
|
|
}
|
|
#endif
|
|
return select(nfds, rd, wr, exc, tv);
|
|
}
|
|
|
|
const char *libtest_arg2 = NULL;
|
|
const char *libtest_arg3 = NULL;
|
|
const char *libtest_arg4 = NULL;
|
|
int test_argc;
|
|
const char **test_argv;
|
|
int testnum;
|
|
|
|
struct curltime tv_test_start; /* for test timing */
|
|
|
|
int unitfail; /* for unittests */
|
|
|
|
int coptind;
|
|
const char *coptarg;
|
|
|
|
int cgetopt(int argc, const char * const argv[], const char *optstring)
|
|
{
|
|
static int optpos = 1;
|
|
int coptopt;
|
|
const char *arg;
|
|
|
|
if(coptind == 0) { /* Reset? */
|
|
coptind = !!argc;
|
|
optpos = 1;
|
|
}
|
|
|
|
arg = argv[coptind];
|
|
if(arg && strcmp(arg, "--") == 0) {
|
|
coptind++;
|
|
return -1;
|
|
}
|
|
else if(!arg || arg[0] != '-') {
|
|
return -1;
|
|
}
|
|
else {
|
|
const char *opt = strchr(optstring, arg[optpos]);
|
|
coptopt = arg[optpos];
|
|
if(!opt) {
|
|
if(!arg[++optpos]) {
|
|
coptind++;
|
|
optpos = 1;
|
|
}
|
|
return '?';
|
|
}
|
|
else if(opt[1] == ':') {
|
|
if(arg[optpos + 1]) {
|
|
coptarg = arg + optpos + 1;
|
|
coptind++;
|
|
optpos = 1;
|
|
return coptopt;
|
|
}
|
|
else if(argv[coptind + 1]) {
|
|
coptarg = argv[coptind + 1];
|
|
coptind += 2;
|
|
optpos = 1;
|
|
return coptopt;
|
|
}
|
|
else {
|
|
if(!arg[++optpos]) {
|
|
coptind++;
|
|
optpos = 1;
|
|
}
|
|
return *optstring == ':' ? ':' : '?';
|
|
}
|
|
}
|
|
else {
|
|
if(!arg[++optpos]) {
|
|
coptind++;
|
|
optpos = 1;
|
|
}
|
|
return coptopt;
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef CURLDEBUG
|
|
static void memory_tracking_init(void)
|
|
{
|
|
char *env;
|
|
/* if CURL_MEMDEBUG is set, this starts memory tracking message logging */
|
|
env = getenv("CURL_MEMDEBUG");
|
|
if(env) {
|
|
/* use the value as file name */
|
|
curl_dbg_memdebug(env);
|
|
}
|
|
/* if CURL_MEMLIMIT is set, this enables fail-on-alloc-number-N feature */
|
|
env = getenv("CURL_MEMLIMIT");
|
|
if(env) {
|
|
char *endptr;
|
|
long num = strtol(env, &endptr, 10);
|
|
if((endptr != env) && (endptr == env + strlen(env)) && (num > 0))
|
|
curl_dbg_memlimit(num);
|
|
}
|
|
}
|
|
#else
|
|
# define memory_tracking_init() Curl_nop_stmt
|
|
#endif
|
|
|
|
/* returns a hexdump in a static memory area */
|
|
char *hexdump(const unsigned char *buf, size_t len)
|
|
{
|
|
static char dump[200 * 3 + 1];
|
|
char *p = dump;
|
|
size_t i;
|
|
if(len > 200)
|
|
return NULL;
|
|
for(i = 0; i < len; i++, p += 3)
|
|
curl_msnprintf(p, 4, "%02x ", buf[i]);
|
|
return dump;
|
|
}
|
|
|
|
#ifndef CURL_DISABLE_WEBSOCKETS
|
|
CURLcode ws_send_ping(CURL *curl, const char *send_payload)
|
|
{
|
|
size_t sent;
|
|
CURLcode result = curl_ws_send(curl, send_payload, strlen(send_payload),
|
|
&sent, 0, CURLWS_PING);
|
|
curl_mfprintf(stderr, "ws: curl_ws_send returned %u, sent %zu\n",
|
|
result, sent);
|
|
return result;
|
|
}
|
|
|
|
CURLcode ws_recv_pong(CURL *curl, const char *expected_payload)
|
|
{
|
|
size_t rlen;
|
|
const struct curl_ws_frame *meta;
|
|
char buffer[256];
|
|
CURLcode result = curl_ws_recv(curl, buffer, sizeof(buffer), &rlen, &meta);
|
|
if(result) {
|
|
curl_mfprintf(stderr, "ws: curl_ws_recv returned %u, received %zd\n",
|
|
result, rlen);
|
|
return result;
|
|
}
|
|
|
|
if(!(meta->flags & CURLWS_PONG)) {
|
|
curl_mfprintf(stderr, "recv_pong: wrong frame, got %zd bytes rflags %x\n",
|
|
rlen, meta->flags);
|
|
return CURLE_RECV_ERROR;
|
|
}
|
|
|
|
curl_mfprintf(stderr, "ws: got PONG back\n");
|
|
if(rlen == strlen(expected_payload) &&
|
|
!memcmp(expected_payload, buffer, rlen)) {
|
|
curl_mfprintf(stderr, "ws: got the same payload back\n");
|
|
return CURLE_OK; /* lib2304 returned 'result' here. Intentional? */
|
|
}
|
|
curl_mfprintf(stderr, "ws: did NOT get the same payload back\n");
|
|
return CURLE_RECV_ERROR;
|
|
}
|
|
|
|
/* just close the connection */
|
|
void ws_close(CURL *curl)
|
|
{
|
|
size_t sent;
|
|
CURLcode result = curl_ws_send(curl, "", 0, &sent, 0, CURLWS_CLOSE);
|
|
curl_mfprintf(stderr, "ws: curl_ws_send returned %u, sent %zu\n",
|
|
result, sent);
|
|
}
|
|
#endif /* CURL_DISABLE_WEBSOCKETS */
|
|
|
|
|
|
int main(int argc, const char **argv)
|
|
{
|
|
const char *URL = "";
|
|
CURLcode result;
|
|
entry_func_t entry_func;
|
|
const char *entry_name;
|
|
char *env;
|
|
size_t tmp;
|
|
|
|
CURLX_SET_BINMODE(stdout);
|
|
|
|
memory_tracking_init();
|
|
#ifdef _WIN32
|
|
curlx_now_init();
|
|
#endif
|
|
|
|
/*
|
|
* Setup proper locale from environment. This is needed to enable locale-
|
|
* specific behavior by the C library in order to test for undesired side
|
|
* effects that could cause in libcurl.
|
|
*/
|
|
#ifdef HAVE_SETLOCALE
|
|
setlocale(LC_ALL, "");
|
|
#endif
|
|
|
|
test_argc = argc - 1;
|
|
test_argv = argv + 1;
|
|
|
|
if(argc < 2) {
|
|
curl_mfprintf(stderr, "Pass testname "
|
|
"(and URL as argument for numbered tests) please\n");
|
|
return 1;
|
|
}
|
|
|
|
entry_name = argv[1];
|
|
entry_func = NULL;
|
|
for(tmp = 0; s_entries[tmp].ptr; ++tmp) {
|
|
if(strcmp(entry_name, s_entries[tmp].name) == 0) {
|
|
entry_func = s_entries[tmp].ptr;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(!entry_func) {
|
|
curl_mfprintf(stderr, "Test '%s' not found.\n", entry_name);
|
|
return 1;
|
|
}
|
|
|
|
if(argc > 2) {
|
|
URL = argv[2];
|
|
curl_mfprintf(stderr, "URL: %s\n", URL);
|
|
}
|
|
|
|
if(argc > 3)
|
|
libtest_arg2 = argv[3];
|
|
|
|
if(argc > 4)
|
|
libtest_arg3 = argv[4];
|
|
|
|
if(argc > 5)
|
|
libtest_arg4 = argv[5];
|
|
|
|
env = getenv("CURL_TESTNUM");
|
|
if(env)
|
|
testnum = atoi(env);
|
|
else
|
|
testnum = 0;
|
|
|
|
result = entry_func(URL);
|
|
curl_mfprintf(stderr, "Test ended with result %d\n", result);
|
|
|
|
#ifdef _WIN32
|
|
/* flush buffers of all streams regardless of mode */
|
|
_flushall();
|
|
#endif
|
|
|
|
/* Regular program status codes are limited to 0..127 and 126 and 127 have
|
|
* special meanings by the shell, so limit a normal return code to 125 */
|
|
return (int)result <= 125 ? (int)result : 125;
|
|
}
|