branch: master
lib2700.c
6473 bytesRaw
/***************************************************************************
 *                                  _   _ ____  _
 *  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"

#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 result = CURLE_OK;
  size_t nsent;

retry:
  result = curl_ws_send(curl, NULL, 0, &nsent, (curl_off_t)size,
                     flags | CURLWS_OFFSET);
  if(result == CURLE_AGAIN) {
    assert(nsent == 0);
    goto retry;
  }
  if(result) {
    curl_mfprintf(stderr, "%s:%d curl_ws_send() failed with code %d (%s)\n",
                  __FILE__, __LINE__, result, curl_easy_strerror(result));
    assert(nsent == 0);
    return result;
  }

  assert(nsent == 0);

  return CURLE_OK;
}

static CURLcode recv_header(CURL *curl, int *flags, curl_off_t *offset,
                            curl_off_t *bytesleft)
{
  CURLcode result = CURLE_OK;
  size_t nread;
  const struct curl_ws_frame *meta;

  *flags = 0;
  *offset = 0;
  *bytesleft = 0;

retry:
  result = curl_ws_recv(curl, NULL, 0, &nread, &meta);
  if(result == CURLE_AGAIN) {
    assert(nread == 0);
    goto retry;
  }
  if(result) {
    curl_mfprintf(stderr, "%s:%d curl_ws_recv() failed with code %d (%s)\n",
                  __FILE__, __LINE__, result, curl_easy_strerror(result));
    assert(nread == 0);
    return result;
  }

  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, " ");

  result = send_header(curl, meta->flags, (size_t)meta->bytesleft);
  if(result)
    return result;

  return CURLE_OK;
}

static CURLcode send_chunk(CURL *curl, int flags, const char *buffer,
                           size_t size, size_t *offset)
{
  CURLcode result = CURLE_OK;
  size_t nsent;

retry:
  result = curl_ws_send(curl, buffer + *offset, size - *offset, &nsent,
                        0, flags);
  if(result == CURLE_AGAIN) {
    assert(nsent == 0);
    goto retry;
  }
  if(result) {
    curl_mfprintf(stderr, "%s:%d curl_ws_send() failed with code %d (%s)\n",
                  __FILE__, __LINE__, result, curl_easy_strerror(result));
    assert(nsent == 0);
    return result;
  }

  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 result = CURLE_OK;
  char buffer[256];
  size_t nread;
  const struct curl_ws_frame *meta;
  size_t sendoffset = 0;

retry:
  result = curl_ws_recv(curl, buffer, sizeof(buffer), &nread, &meta);
  if(result == CURLE_AGAIN) {
    assert(nread == 0);
    goto retry;
  }
  if(result) {
    curl_mfprintf(stderr, "%s:%d curl_ws_recv() failed with code %d (%s)\n",
                  __FILE__, __LINE__, result, curl_easy_strerror(result));
    assert(nread == 0);
    return result;
  }

  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) {
    result = send_chunk(curl, flags, buffer, nread, &sendoffset);
    if(result)
      return result;
  }

  return CURLE_OK;
}

static CURLcode recv_frame(CURL *curl, bool *stop)
{
  CURLcode result = CURLE_OK;
  int flags = 0;
  curl_off_t offset = 0;
  curl_off_t bytesleft = 0;

  result = recv_header(curl, &flags, &offset, &bytesleft);
  if(result)
    return result;

  while(bytesleft > 0) {
    result = recv_chunk(curl, flags, &offset, &bytesleft);
    if(result)
      return result;
  }

  if(flags & CURLWS_CLOSE)
    *stop = true;

  curl_mfprintf(stdout, "\n");

  return result;
}
#endif

static CURLcode test_lib2700(const char *URL)
{
#ifndef CURL_DISABLE_WEBSOCKETS
  CURLcode result = 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);

  result = curl_easy_perform(curl);
  if(result) {
    curl_mfprintf(stderr,
                  "%s:%d curl_easy_perform() failed with code %d (%s)\n",
                  __FILE__, __LINE__, result, curl_easy_strerror(result));
    goto test_cleanup;
  }

  while(!stop) {
    result = recv_frame(curl, &stop);
    if(result)
      goto test_cleanup;
  }

test_cleanup:
  curl_easy_cleanup(curl);
  curl_global_cleanup();
  return result;
#else
  NO_SUPPORT_BUILT_IN
#endif
}