branch: master
first.h
24081 bytesRaw
#ifndef HEADER_LIBTEST_FIRST_H
#define HEADER_LIBTEST_FIRST_H
/***************************************************************************
 *                                  _   _ ____  _
 *  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
 *
 ***************************************************************************/
#define CURL_NO_OLDIES
#define CURL_DISABLE_DEPRECATION

/* Now include the curl_setup.h file from libcurl's private libdir (the source
   version, but that might include "curl_config.h" from the build directory so
   we need both of them in the include path), so that we get good in-depth
   knowledge about the system we are building this on */
#include "curl_setup.h"

typedef CURLcode (*entry_func_t)(const char *);

struct entry_s {
  const char *name;
  entry_func_t ptr;
};

extern const struct entry_s s_entries[];

extern int unitfail; /* for unittests */

#ifdef UNITTESTS
#include "unitprotos.h"
#endif

#include "curlx/base64.h" /* for curlx_base64* */
#include "curlx/dynbuf.h" /* for curlx_dyn_*() */
#include "curlx/fopen.h" /* for curlx_f*() */
#include "curlx/strcopy.h" /* for curlx_strcopy() */
#include "curlx/strerr.h" /* for curlx_strerror() */
#include "curlx/strparse.h" /* for curlx_str_* parsing functions */
#include "curlx/timediff.h" /* for timediff_t type and related functions */
#include "curlx/timeval.h" /* for curlx_now type and related functions */
#include "curlx/wait.h" /* for curlx_wait_ms() */

#ifdef HAVE_SYS_SELECT_H
/* since so many tests use select(), we can just as well include it here */
#include <sys/select.h>
#endif

#define test_setopt(A, B, C)            \
  do {                                  \
    result = curl_easy_setopt(A, B, C); \
    if(result != CURLE_OK)              \
      goto test_cleanup;                \
  } while(0)

#define test_multi_setopt(A, B, C)       \
  do {                                   \
    result = curl_multi_setopt(A, B, C); \
    if(result != CURLE_OK)               \
      goto test_cleanup;                 \
  } while(0)

extern const char *libtest_arg2; /* set by first.c to the argv[2] or NULL */
extern const char *libtest_arg3; /* set by first.c to the argv[3] or NULL */
extern const char *libtest_arg4; /* set by first.c to the argv[4] or NULL */

/* argc and argv as passed in to the main() function */
extern int test_argc;
extern const char **test_argv;
extern int testnum;
extern struct curltime tv_test_start; /* for test timing */

extern int coptind;
extern const char *coptarg;
int cgetopt(int argc, const char * const argv[], const char *optstring);

extern int select_wrapper(int nfds, fd_set *rd, fd_set *wr, fd_set *exc,
                          struct timeval *tv);

extern char *hexdump(const unsigned char *buf, size_t len);

#ifndef CURL_DISABLE_WEBSOCKETS
CURLcode ws_send_ping(CURL *curl, const char *send_payload);
CURLcode ws_recv_pong(CURL *curl, const char *expected_payload);
void ws_close(CURL *curl);  /* just close the connection */
#endif

/*
 * TEST_ERR_* values must within the CURLcode range to not cause compiler
 * errors.
 *
 * For portability reasons TEST_ERR_* values should be less than 127.
 */

#define TEST_ERR_MAJOR_BAD    CURLE_OBSOLETE20
#define TEST_ERR_RUNS_FOREVER CURLE_OBSOLETE24
#define TEST_ERR_EASY_INIT    CURLE_OBSOLETE29
#define TEST_ERR_MULTI        CURLE_OBSOLETE32
#define TEST_ERR_NUM_HANDLES  CURLE_OBSOLETE34
#define TEST_ERR_SELECT       CURLE_OBSOLETE40
#define TEST_ERR_SUCCESS      CURLE_OBSOLETE41
#define TEST_ERR_FAILURE      CURLE_OBSOLETE44
#define TEST_ERR_USAGE        CURLE_OBSOLETE46
#define TEST_ERR_FOPEN        CURLE_OBSOLETE50
#define TEST_ERR_FSTAT        CURLE_OBSOLETE51
#define TEST_ERR_BAD_TIMEOUT  CURLE_OBSOLETE57

/*
 * Macros for test source code readability/maintainability.
 *
 * All of the following macros require that an int data type 'res' variable
 * exists in scope where macro is used, and that it has been initialized to
 * zero before the macro is used.
 *
 * exe_* and chk_* macros are helper macros not intended to be used from
 * outside of this header file. Arguments 'Y' and 'Z' of these represent
 * source code file and line number, while Arguments 'A', 'B', etc, are
 * the arguments used to actually call a libcurl function.
 *
 * All easy_* and multi_* macros call a libcurl function and evaluate if
 * the function has succeeded or failed. When the function succeeds 'res'
 * variable is not set nor cleared and program continues normal flow. On
 * the other hand if function fails 'res' variable is set and a jump to
 * label 'test_cleanup' is performed.
 *
 * Every easy_* and multi_* macros have a res_easy_* and res_multi_* macro
 * counterpart that operates in the same way with the exception that no
 * jump takes place in case of failure. res_easy_* and res_multi_* macros
 * should be immediately followed by checking if 'res' variable has been
 * set.
 *
 * 'res' variable when set will hold a CURLcode, CURLMcode, or any of the
 * TEST_ERR_* values defined above. It is advisable to return this value
 * as test result.
 */

/* ---------------------------------------------------------------- */

#define exe_easy_init(A, Y, Z)                                        \
  do {                                                                \
    (A) = curl_easy_init();                                           \
    if(!(A)) {                                                        \
      curl_mfprintf(stderr, "%s:%d curl_easy_init() failed\n", Y, Z); \
      result = TEST_ERR_EASY_INIT;                                    \
    }                                                                 \
  } while(0)

#define res_easy_init(A) \
  exe_easy_init(A, __FILE__, __LINE__)

#define chk_easy_init(A, Y, Z) \
  do {                         \
    exe_easy_init(A, Y, Z);    \
    if(result)                 \
      goto test_cleanup;       \
  } while(0)

#define easy_init(A) \
  chk_easy_init(A, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_multi_init(A, Y, Z)                                        \
  do {                                                                 \
    (A) = curl_multi_init();                                           \
    if(!(A)) {                                                         \
      curl_mfprintf(stderr, "%s:%d curl_multi_init() failed\n", Y, Z); \
      result = TEST_ERR_MULTI;                                         \
    }                                                                  \
  } while(0)

#define res_multi_init(A) \
  exe_multi_init(A, __FILE__, __LINE__)

#define chk_multi_init(A, Y, Z) \
  do {                          \
    exe_multi_init(A, Y, Z);    \
    if(result)                  \
      goto test_cleanup;        \
  } while(0)

#define multi_init(A) \
  chk_multi_init(A, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_easy_setopt(A, B, C, Y, Z)                  \
  do {                                                  \
    CURLcode ec = curl_easy_setopt(A, B, C);            \
    if(ec != CURLE_OK) {                                \
      curl_mfprintf(stderr,                             \
                    "%s:%d curl_easy_setopt() failed, " \
                    "with code %d (%s)\n",              \
                    Y, Z, ec, curl_easy_strerror(ec));  \
      result = ec;                                      \
    }                                                   \
  } while(0)

#define res_easy_setopt(A, B, C) \
  exe_easy_setopt(A, B, C, __FILE__, __LINE__)

#define chk_easy_setopt(A, B, C, Y, Z) \
  do {                                 \
    exe_easy_setopt(A, B, C, Y, Z);    \
    if(result)                         \
      goto test_cleanup;               \
  } while(0)

#define easy_setopt(A, B, C) \
  chk_easy_setopt(A, B, C, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_multi_setopt(A, B, C, Y, Z)                  \
  do {                                                   \
    CURLMcode ec = curl_multi_setopt(A, B, C);           \
    if(ec != CURLM_OK) {                                 \
      curl_mfprintf(stderr,                              \
                    "%s:%d curl_multi_setopt() failed, " \
                    "with code %d (%s)\n",               \
                    Y, Z, ec, curl_multi_strerror(ec));  \
      result = TEST_ERR_MULTI;                           \
    }                                                    \
  } while(0)

#define res_multi_setopt(A, B, C) \
  exe_multi_setopt(A, B, C, __FILE__, __LINE__)

#define chk_multi_setopt(A, B, C, Y, Z) \
  do {                                  \
    exe_multi_setopt(A, B, C, Y, Z);    \
    if(result)                          \
      goto test_cleanup;                \
  } while(0)

#define multi_setopt(A, B, C) \
  chk_multi_setopt(A, B, C, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_multi_add_handle(A, B, Y, Z)                     \
  do {                                                       \
    CURLMcode ec = curl_multi_add_handle(A, B);              \
    if(ec != CURLM_OK) {                                     \
      curl_mfprintf(stderr,                                  \
                    "%s:%d curl_multi_add_handle() failed, " \
                    "with code %d (%s)\n",                   \
                    Y, Z, ec, curl_multi_strerror(ec));      \
      result = TEST_ERR_MULTI;                               \
    }                                                        \
  } while(0)

#define res_multi_add_handle(A, B) \
  exe_multi_add_handle(A, B, __FILE__, __LINE__)

#define chk_multi_add_handle(A, B, Y, Z) \
  do {                                   \
    exe_multi_add_handle(A, B, Y, Z);    \
    if(result)                           \
      goto test_cleanup;                 \
  } while(0)

#define multi_add_handle(A, B) \
  chk_multi_add_handle(A, B, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_multi_remove_handle(A, B, Y, Z)                     \
  do {                                                          \
    CURLMcode ec = curl_multi_remove_handle(A, B);              \
    if(ec != CURLM_OK) {                                        \
      curl_mfprintf(stderr,                                     \
                    "%s:%d curl_multi_remove_handle() failed, " \
                    "with code %d (%s)\n",                      \
                    Y, Z, ec, curl_multi_strerror(ec));         \
      result = TEST_ERR_MULTI;                                  \
    }                                                           \
  } while(0)

#define res_multi_remove_handle(A, B) \
  exe_multi_remove_handle(A, B, __FILE__, __LINE__)

#define chk_multi_remove_handle(A, B, Y, Z) \
  do {                                      \
    exe_multi_remove_handle(A, B, Y, Z);    \
    if(result)                              \
      goto test_cleanup;                    \
  } while(0)

#define multi_remove_handle(A, B) \
  chk_multi_remove_handle(A, B, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_multi_perform(A, B, Y, Z)                                    \
  do {                                                                   \
    CURLMcode ec = curl_multi_perform(A, B);                             \
    if(ec != CURLM_OK) {                                                 \
      curl_mfprintf(stderr,                                              \
                    "%s:%d curl_multi_perform() failed, "                \
                    "with code %d (%s)\n",                               \
                    Y, Z, ec, curl_multi_strerror(ec));                  \
      result = TEST_ERR_MULTI;                                           \
    }                                                                    \
    else if(*(B) < 0) {                                                  \
      curl_mfprintf(stderr,                                              \
                    "%s:%d curl_multi_perform() succeeded, "             \
                    "but returned invalid running_handles value (%d)\n", \
                    Y, Z, (int)*(B));                                    \
      result = TEST_ERR_NUM_HANDLES;                                     \
    }                                                                    \
  } while(0)

#define res_multi_perform(A, B) \
  exe_multi_perform(A, B, __FILE__, __LINE__)

#define chk_multi_perform(A, B, Y, Z) \
  do {                                \
    exe_multi_perform(A, B, Y, Z);    \
    if(result)                        \
      goto test_cleanup;              \
  } while(0)

#define multi_perform(A, B) \
  chk_multi_perform(A, B, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_multi_fdset(A, B, C, D, E, Y, Z)                    \
  do {                                                          \
    CURLMcode ec = curl_multi_fdset(A, B, C, D, E);             \
    if(ec != CURLM_OK) {                                        \
      curl_mfprintf(stderr,                                     \
                    "%s:%d curl_multi_fdset() failed, "         \
                    "with code %d (%s)\n",                      \
                    Y, Z, ec, curl_multi_strerror(ec));         \
      result = TEST_ERR_MULTI;                                  \
    }                                                           \
    else if(*(E) < -1) {                                        \
      curl_mfprintf(stderr,                                     \
                    "%s:%d curl_multi_fdset() succeeded, "      \
                    "but returned invalid max_fd value (%d)\n", \
                    Y, Z, (int)*(E));                           \
      result = TEST_ERR_NUM_HANDLES;                            \
    }                                                           \
  } while(0)

#define res_multi_fdset(A, B, C, D, E) \
  exe_multi_fdset(A, B, C, D, E, __FILE__, __LINE__)

#define chk_multi_fdset(A, B, C, D, E, Y, Z) \
  do {                                       \
    exe_multi_fdset(A, B, C, D, E, Y, Z);    \
    if(result)                               \
      goto test_cleanup;                     \
  } while(0)

#define multi_fdset(A, B, C, D, E) \
  chk_multi_fdset(A, B, C, D, E, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_multi_timeout(A, B, Y, Z)                             \
  do {                                                            \
    CURLMcode ec = curl_multi_timeout(A, B);                      \
    if(ec != CURLM_OK) {                                          \
      curl_mfprintf(stderr,                                       \
                    "%s:%d curl_multi_timeout() failed, "         \
                    "with code %d (%s)\n",                        \
                    Y, Z, ec, curl_multi_strerror(ec));           \
      result = TEST_ERR_BAD_TIMEOUT;                              \
    }                                                             \
    else if(*(B) < -1L) {                                         \
      curl_mfprintf(stderr,                                       \
                    "%s:%d curl_multi_timeout() succeeded, "      \
                    "but returned invalid timeout value (%ld)\n", \
                    Y, Z, (long)*(B));                            \
      result = TEST_ERR_BAD_TIMEOUT;                              \
    }                                                             \
  } while(0)

#define res_multi_timeout(A, B) \
  exe_multi_timeout(A, B, __FILE__, __LINE__)

#define chk_multi_timeout(A, B, Y, Z) \
  do {                                \
    exe_multi_timeout(A, B, Y, Z);    \
    if(result)                        \
      goto test_cleanup;              \
  } while(0)

#define multi_timeout(A, B) \
  chk_multi_timeout(A, B, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_multi_poll(A, B, C, D, E, Y, Z)                     \
  do {                                                          \
    CURLMcode ec = curl_multi_poll(A, B, C, D, E);              \
    if(ec != CURLM_OK) {                                        \
      curl_mfprintf(stderr,                                     \
                    "%s:%d curl_multi_poll() failed, "          \
                    "with code %d (%s)\n",                      \
                    Y, Z, ec, curl_multi_strerror(ec));         \
      result = TEST_ERR_MULTI;                                  \
    }                                                           \
    else if(*(E) < 0) {                                         \
      curl_mfprintf(stderr,                                     \
                    "%s:%d curl_multi_poll() succeeded, "       \
                    "but returned invalid numfds value (%d)\n", \
                    Y, Z, (int)*(E));                           \
      result = TEST_ERR_NUM_HANDLES;                            \
    }                                                           \
  } while(0)

#define res_multi_poll(A, B, C, D, E) \
  exe_multi_poll(A, B, C, D, E, __FILE__, __LINE__)

#define chk_multi_poll(A, B, C, D, E, Y, Z) \
  do {                                      \
    exe_multi_poll(A, B, C, D, E, Y, Z);    \
    if(result)                              \
      goto test_cleanup;                    \
  } while(0)

#define multi_poll(A, B, C, D, E) \
  chk_multi_poll(A, B, C, D, E, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_multi_wakeup(A, Y, Z)                        \
  do {                                                   \
    CURLMcode ec = curl_multi_wakeup(A);                 \
    if(ec != CURLM_OK) {                                 \
      curl_mfprintf(stderr,                              \
                    "%s:%d curl_multi_wakeup() failed, " \
                    "with code %d (%s)\n",               \
                    Y, Z, ec, curl_multi_strerror(ec));  \
      result = TEST_ERR_MULTI;                           \
    }                                                    \
  } while(0)

#define res_multi_wakeup(A) \
  exe_multi_wakeup(A, __FILE__, __LINE__)

#define chk_multi_wakeup(A, Y, Z) \
  do {                            \
    exe_multi_wakeup(A, Y, Z);    \
    if(result)                    \
      goto test_cleanup;          \
  } while(0)

#define multi_wakeup(A) \
  chk_multi_wakeup(A, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_select_test(A, B, C, D, E, Y, Z)                             \
  do {                                                                   \
    if(select_wrapper(A, B, C, D, E) == -1) {                            \
      int ec = SOCKERRNO;                                                \
      char ecbuf[STRERROR_LEN];                                          \
      curl_mfprintf(stderr,                                              \
                    "%s:%d select() failed, with "                       \
                    "errno %d (%s)\n",                                   \
                    Y, Z, ec, curlx_strerror(ec, ecbuf, sizeof(ecbuf))); \
      result = TEST_ERR_SELECT;                                          \
    }                                                                    \
  } while(0)

#define res_select_test(A, B, C, D, E) \
  exe_select_test(A, B, C, D, E, __FILE__, __LINE__)

#define chk_select_test(A, B, C, D, E, Y, Z) \
  do {                                       \
    exe_select_test(A, B, C, D, E, Y, Z);    \
    if(result)                               \
      goto test_cleanup;                     \
  } while(0)

#define select_test(A, B, C, D, E) \
  chk_select_test(A, B, C, D, E, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define start_test_timing()      \
  do {                           \
    tv_test_start = curlx_now(); \
  } while(0)

#define TEST_HANG_TIMEOUT (60 * 1000)  /* global default */

#define exe_test_timedout(T, Y, Z)                                        \
  do {                                                                    \
    timediff_t timediff = curlx_timediff_ms(curlx_now(), tv_test_start);  \
    if(timediff > (T)) {                                                  \
      curl_mfprintf(stderr,                                               \
                    "%s:%d ABORTING TEST, since it seems "                \
                    "that it would have run forever (%ld ms > %ld ms)\n", \
                    Y, Z, (long)timediff, (long)TEST_HANG_TIMEOUT);       \
      result = TEST_ERR_RUNS_FOREVER;                                     \
    }                                                                     \
  } while(0)

#define res_test_timedout() \
  exe_test_timedout(TEST_HANG_TIMEOUT, __FILE__, __LINE__)

#define res_test_timedout_custom(T) \
  exe_test_timedout(T, __FILE__, __LINE__)

#define chk_test_timedout(T, Y, Z) \
  do {                             \
    exe_test_timedout(T, Y, Z);    \
    if(result)                     \
      goto test_cleanup;           \
  } while(0)

#define abort_on_test_timeout() \
  chk_test_timedout(TEST_HANG_TIMEOUT, __FILE__, __LINE__)

#define abort_on_test_timeout_custom(T) \
  chk_test_timedout(T, __FILE__, __LINE__)

/* ---------------------------------------------------------------- */

#define exe_global_init(A, Y, Z)                        \
  do {                                                  \
    CURLcode ec = curl_global_init(A);                  \
    if(ec != CURLE_OK) {                                \
      curl_mfprintf(stderr,                             \
                    "%s:%d curl_global_init() failed, " \
                    "with code %d (%s)\n",              \
                    Y, Z, ec, curl_easy_strerror(ec));  \
      result = ec;                                      \
    }                                                   \
  } while(0)

#define res_global_init(A) \
  exe_global_init(A, __FILE__, __LINE__)

#define chk_global_init(A, Y, Z) \
  do {                           \
    exe_global_init(A, Y, Z);    \
    if(result)                   \
      return result;             \
  } while(0)

/* global_init() is different than other macros. In case of
   failure it 'return's instead of going to 'test_cleanup'. */

#define global_init(A) \
  chk_global_init(A, __FILE__, __LINE__)

#define NO_SUPPORT_BUILT_IN                     \
  {                                             \
    (void)URL;                                  \
    curl_mfprintf(stderr, "Missing support\n"); \
    return CURLE_UNSUPPORTED_PROTOCOL;          \
  }

#define NUM_HANDLES 4  /* global default */

#endif /* HEADER_LIBTEST_FIRST_H */