Blame src/testcurl/test_timeout.c

Packit 875988
/*
Packit 875988
     This file is part of libmicrohttpd
Packit 875988
     Copyright (C) 2007 Christian Grothoff
Packit 875988
Packit 875988
     libmicrohttpd is free software; you can redistribute it and/or modify
Packit 875988
     it under the terms of the GNU General Public License as published
Packit 875988
     by the Free Software Foundation; either version 2, or (at your
Packit 875988
     option) any later version.
Packit 875988
Packit 875988
     libmicrohttpd is distributed in the hope that it will be useful, but
Packit 875988
     WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 875988
     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 875988
     General Public License for more details.
Packit 875988
Packit 875988
     You should have received a copy of the GNU General Public License
Packit 875988
     along with libmicrohttpd; see the file COPYING.  If not, write to the
Packit 875988
     Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Packit 875988
     Boston, MA 02110-1301, USA.
Packit 875988
*/
Packit 875988
Packit 875988
/**
Packit 875988
 * @file test_timeout.c
Packit 875988
 * @brief  Testcase for libmicrohttpd PUT operations
Packit 875988
 * @author Matthias Wachs
Packit 875988
 */
Packit 875988
Packit 875988
#include "MHD_config.h"
Packit 875988
#include "platform.h"
Packit 875988
#include <curl/curl.h>
Packit 875988
#include <microhttpd.h>
Packit 875988
#include <stdlib.h>
Packit 875988
#include <string.h>
Packit 875988
#include <time.h>
Packit 875988
Packit 875988
#ifndef WINDOWS
Packit 875988
#include <unistd.h>
Packit 875988
#endif
Packit 875988
Packit 875988
static int oneone;
Packit 875988
Packit 875988
static int withTimeout = 1;
Packit 875988
Packit 875988
static int withoutTimeout = 1;
Packit 875988
Packit 875988
struct CBC
Packit 875988
{
Packit 875988
  char *buf;
Packit 875988
  size_t pos;
Packit 875988
  size_t size;
Packit 875988
};
Packit 875988
Packit 875988
Packit 875988
static void 
Packit 875988
termination_cb (void *cls,
Packit 875988
		struct MHD_Connection *connection,
Packit 875988
		void **con_cls,
Packit 875988
		enum MHD_RequestTerminationCode toe)
Packit 875988
{
Packit 875988
  int *test = cls;
Packit 875988
  (void)connection;(void)con_cls;       /* Unused. Silent compiler warning. */
Packit 875988
Packit 875988
  switch (toe)
Packit 875988
    {
Packit 875988
    case MHD_REQUEST_TERMINATED_COMPLETED_OK :
Packit 875988
      if (test == &withoutTimeout)
Packit 875988
	{
Packit 875988
	  withoutTimeout = 0;
Packit 875988
	}
Packit 875988
      break;
Packit 875988
    case MHD_REQUEST_TERMINATED_WITH_ERROR :
Packit 875988
    case MHD_REQUEST_TERMINATED_READ_ERROR :
Packit 875988
      break;
Packit 875988
    case MHD_REQUEST_TERMINATED_TIMEOUT_REACHED :
Packit 875988
      if (test == &withTimeout)
Packit 875988
	{
Packit 875988
	  withTimeout = 0;
Packit 875988
	}
Packit 875988
      break;
Packit 875988
    case MHD_REQUEST_TERMINATED_DAEMON_SHUTDOWN:
Packit 875988
      break;
Packit 875988
    case MHD_REQUEST_TERMINATED_CLIENT_ABORT:
Packit 875988
      break;
Packit 875988
    }
Packit 875988
}
Packit 875988
Packit 875988
Packit 875988
static size_t
Packit 875988
putBuffer (void *stream, size_t size, size_t nmemb, void *ptr)
Packit 875988
{
Packit 875988
  unsigned int *pos = ptr;
Packit 875988
  unsigned int wrt;
Packit 875988
Packit 875988
  wrt = size * nmemb;
Packit 875988
  if (wrt > 8 - (*pos))
Packit 875988
	wrt = 8 - (*pos);
Packit 875988
  memcpy (stream, &("Hello123"[*pos]), wrt);
Packit 875988
  (*pos) += wrt;
Packit 875988
  return wrt;
Packit 875988
}
Packit 875988
Packit 875988
Packit 875988
static size_t
Packit 875988
putBuffer_fail (void *stream, size_t size, size_t nmemb, void *ptr)
Packit 875988
{
Packit 875988
  (void)stream;(void)size;(void)nmemb;(void)ptr;        /* Unused. Silent compiler warning. */
Packit 875988
  return 0;
Packit 875988
}
Packit 875988
Packit 875988
Packit 875988
static size_t
Packit 875988
copyBuffer (void *ptr, size_t size, size_t nmemb, void *ctx)
Packit 875988
{
Packit 875988
  struct CBC *cbc = ctx;
Packit 875988
Packit 875988
  if (cbc->pos + size * nmemb > cbc->size)
Packit 875988
    return 0;                   /* overflow */
Packit 875988
  memcpy (&cbc->buf[cbc->pos], ptr, size * nmemb);
Packit 875988
  cbc->pos += size * nmemb;
Packit 875988
  return size * nmemb;
Packit 875988
}
Packit 875988
Packit 875988
Packit 875988
static int
Packit 875988
ahc_echo (void *cls,
Packit 875988
          struct MHD_Connection *connection,
Packit 875988
          const char *url,
Packit 875988
          const char *method,
Packit 875988
          const char *version,
Packit 875988
          const char *upload_data, size_t *upload_data_size,
Packit 875988
          void **unused)
Packit 875988
{
Packit 875988
  int *done = cls;
Packit 875988
  struct MHD_Response *response;
Packit 875988
  int ret;
Packit 875988
  (void)version;(void)unused;   /* Unused. Silent compiler warning. */
Packit 875988
Packit 875988
  if (0 != strcmp ("PUT", method))
Packit 875988
    return MHD_NO;              /* unexpected method */
Packit 875988
  if ((*done) == 0)
Packit 875988
    {
Packit 875988
      if (*upload_data_size != 8)
Packit 875988
        return MHD_YES;         /* not yet ready */
Packit 875988
      if (0 == memcmp (upload_data, "Hello123", 8))
Packit 875988
        {
Packit 875988
          *upload_data_size = 0;
Packit 875988
        }
Packit 875988
      else
Packit 875988
        {
Packit 875988
          printf ("Invalid upload data `%8s'!\n", upload_data);
Packit 875988
          return MHD_NO;
Packit 875988
        }
Packit 875988
      *done = 1;
Packit 875988
      return MHD_YES;
Packit 875988
    }
Packit 875988
  response = MHD_create_response_from_buffer (strlen (url),
Packit 875988
					      (void *) url, 
Packit 875988
					      MHD_RESPMEM_MUST_COPY);
Packit 875988
  ret = MHD_queue_response (connection, MHD_HTTP_OK, response);
Packit 875988
  MHD_destroy_response (response);
Packit 875988
  return ret;
Packit 875988
}
Packit 875988
Packit 875988
Packit 875988
static int
Packit 875988
testWithoutTimeout ()
Packit 875988
{
Packit 875988
  struct MHD_Daemon *d;
Packit 875988
  CURL *c;
Packit 875988
  char buf[2048];
Packit 875988
  struct CBC cbc;
Packit 875988
  unsigned int pos = 0;
Packit 875988
  int done_flag = 0;
Packit 875988
  CURLcode errornum;
Packit 875988
  int port;
Packit 875988
Packit 875988
  if (MHD_NO != MHD_is_feature_supported (MHD_FEATURE_AUTODETECT_BIND_PORT))
Packit 875988
    port = 0;
Packit 875988
  else
Packit 875988
    {
Packit 875988
      port = 1500;
Packit 875988
      if (oneone)
Packit 875988
        port += 5;
Packit 875988
    }
Packit 875988
Packit 875988
  cbc.buf = buf;
Packit 875988
  cbc.size = 2048;
Packit 875988
  cbc.pos = 0;
Packit 875988
  d = MHD_start_daemon (MHD_USE_INTERNAL_POLLING_THREAD | MHD_USE_ERROR_LOG,
Packit 875988
                        port,
Packit 875988
                        NULL, NULL, &ahc_echo, &done_flag,
Packit 875988
                        MHD_OPTION_CONNECTION_TIMEOUT, 2,
Packit 875988
                        MHD_OPTION_NOTIFY_COMPLETED, &termination_cb, &withTimeout,
Packit 875988
                        MHD_OPTION_END);
Packit 875988
  if (d == NULL)
Packit 875988
    return 1;
Packit 875988
  if (0 == port)
Packit 875988
    {
Packit 875988
      const union MHD_DaemonInfo *dinfo;
Packit 875988
      dinfo = MHD_get_daemon_info (d, MHD_DAEMON_INFO_BIND_PORT);
Packit 875988
      if (NULL == dinfo || 0 == dinfo->port)
Packit 875988
        { MHD_stop_daemon (d); return 32; }
Packit 875988
      port = (int)dinfo->port;
Packit 875988
    }
Packit 875988
  c = curl_easy_init ();
Packit 875988
  curl_easy_setopt (c, CURLOPT_URL, "http://127.0.0.1/hello_world");
Packit 875988
  curl_easy_setopt (c, CURLOPT_PORT, (long)port);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEFUNCTION, &copyBuffer);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEDATA, &cbc);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READFUNCTION, &putBuffer);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READDATA, &pos;;
Packit 875988
  curl_easy_setopt (c, CURLOPT_UPLOAD, 1L);
Packit 875988
  curl_easy_setopt (c, CURLOPT_INFILESIZE_LARGE, (curl_off_t) 8L);
Packit 875988
  curl_easy_setopt (c, CURLOPT_FAILONERROR, 1);
Packit 875988
  curl_easy_setopt (c, CURLOPT_TIMEOUT, 150L);
Packit 875988
  if (oneone)
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1);
Packit 875988
  else
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_0);
Packit 875988
  curl_easy_setopt (c, CURLOPT_CONNECTTIMEOUT, 150L);
Packit 875988
  /* NOTE: use of CONNECTTIMEOUT without also
Packit 875988
   *   setting NOSIGNAL results in really weird
Packit 875988
   *   crashes on my system! */
Packit 875988
  curl_easy_setopt (c, CURLOPT_NOSIGNAL, 1);
Packit 875988
  if (CURLE_OK != (errornum = curl_easy_perform (c)))
Packit 875988
    {
Packit 875988
      curl_easy_cleanup (c);
Packit 875988
      MHD_stop_daemon (d);
Packit 875988
      return 2;
Packit 875988
    }
Packit 875988
  curl_easy_cleanup (c);
Packit 875988
  MHD_stop_daemon (d);
Packit 875988
  if (cbc.pos != strlen ("/hello_world"))
Packit 875988
    return 4;
Packit 875988
  if (0 != strncmp ("/hello_world", cbc.buf, strlen ("/hello_world")))
Packit 875988
    return 8;
Packit 875988
  return 0;
Packit 875988
}
Packit 875988
Packit 875988
static int
Packit 875988
testWithTimeout ()
Packit 875988
{
Packit 875988
  struct MHD_Daemon *d;
Packit 875988
  CURL *c;
Packit 875988
  char buf[2048];
Packit 875988
  struct CBC cbc;
Packit 875988
  int done_flag = 0;
Packit 875988
  CURLcode errornum;
Packit 875988
  int port;
Packit 875988
Packit 875988
  if (MHD_NO != MHD_is_feature_supported (MHD_FEATURE_AUTODETECT_BIND_PORT))
Packit 875988
    port = 0;
Packit 875988
  else
Packit 875988
    {
Packit 875988
      port = 1501;
Packit 875988
      if (oneone)
Packit 875988
        port += 5;
Packit 875988
    }
Packit 875988
Packit 875988
  cbc.buf = buf;
Packit 875988
  cbc.size = 2048;
Packit 875988
  cbc.pos = 0;
Packit 875988
  d = MHD_start_daemon (MHD_USE_INTERNAL_POLLING_THREAD | MHD_USE_ERROR_LOG,
Packit 875988
                        port,
Packit 875988
                        NULL, NULL, &ahc_echo, &done_flag,
Packit 875988
                        MHD_OPTION_CONNECTION_TIMEOUT, 2,
Packit 875988
                        MHD_OPTION_NOTIFY_COMPLETED, &termination_cb, &withoutTimeout,
Packit 875988
                        MHD_OPTION_END);
Packit 875988
  if (d == NULL)
Packit 875988
    return 16;
Packit 875988
  if (0 == port)
Packit 875988
    {
Packit 875988
      const union MHD_DaemonInfo *dinfo;
Packit 875988
      dinfo = MHD_get_daemon_info (d, MHD_DAEMON_INFO_BIND_PORT);
Packit 875988
      if (NULL == dinfo || 0 == dinfo->port)
Packit 875988
        { MHD_stop_daemon (d); return 32; }
Packit 875988
      port = (int)dinfo->port;
Packit 875988
    }
Packit 875988
  c = curl_easy_init ();
Packit 875988
  curl_easy_setopt (c, CURLOPT_URL, "http://127.0.0.1/hello_world");
Packit 875988
  curl_easy_setopt (c, CURLOPT_PORT, (long)port);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEFUNCTION, &copyBuffer);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEDATA, &cbc);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READFUNCTION, &putBuffer_fail);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READDATA, &testWithTimeout);
Packit 875988
  curl_easy_setopt (c, CURLOPT_UPLOAD, 1L);
Packit 875988
  curl_easy_setopt (c, CURLOPT_INFILESIZE_LARGE, (curl_off_t) 8L);
Packit 875988
  curl_easy_setopt (c, CURLOPT_FAILONERROR, 1);
Packit 875988
  curl_easy_setopt (c, CURLOPT_TIMEOUT, 150L);
Packit 875988
  if (oneone)
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1);
Packit 875988
  else
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_0);
Packit 875988
  curl_easy_setopt (c, CURLOPT_CONNECTTIMEOUT, 150L);
Packit 875988
  /* NOTE: use of CONNECTTIMEOUT without also
Packit 875988
   *   setting NOSIGNAL results in really weird
Packit 875988
   *   crashes on my system! */
Packit 875988
  curl_easy_setopt (c, CURLOPT_NOSIGNAL, 1);
Packit 875988
  if (CURLE_OK != (errornum = curl_easy_perform (c)))
Packit 875988
    {
Packit 875988
      curl_easy_cleanup (c);
Packit 875988
      MHD_stop_daemon (d);
Packit 875988
      if (errornum == CURLE_GOT_NOTHING)
Packit 875988
    	  /* mhd had the timeout */
Packit 875988
    	  return 0;
Packit 875988
      else
Packit 875988
    	  /* curl had the timeout first */
Packit 875988
    	  return 32;
Packit 875988
    }
Packit 875988
  curl_easy_cleanup (c);
Packit 875988
  MHD_stop_daemon (d);
Packit 875988
  return 64;
Packit 875988
}
Packit 875988
Packit 875988
Packit 875988
int
Packit 875988
main (int argc, char *const *argv)
Packit 875988
{
Packit 875988
  unsigned int errorCount = 0;
Packit 875988
  (void)argc;   /* Unused. Silent compiler warning. */
Packit 875988
Packit 875988
  oneone = (NULL != strrchr (argv[0], (int) '/')) ?
Packit 875988
    (NULL != strstr (strrchr (argv[0], (int) '/'), "11")) : 0;
Packit 875988
  if (0 != curl_global_init (CURL_GLOBAL_WIN32))
Packit 875988
    return 16;
Packit 875988
  errorCount += testWithoutTimeout ();
Packit 875988
  errorCount += testWithTimeout ();
Packit 875988
  if (errorCount != 0)
Packit 875988
    fprintf (stderr, 
Packit 875988
	     "Error during test execution (code: %u)\n",
Packit 875988
	     errorCount);
Packit 875988
  curl_global_cleanup ();
Packit 875988
  if ((withTimeout == 0) && (withoutTimeout == 0))
Packit 875988
    return 0;
Packit 875988
  else
Packit 875988
    return errorCount;       /* 0 == pass */
Packit 875988
}