113498266Sopenharmony_ci/*************************************************************************** 213498266Sopenharmony_ci * _ _ ____ _ 313498266Sopenharmony_ci * Project ___| | | | _ \| | 413498266Sopenharmony_ci * / __| | | | |_) | | 513498266Sopenharmony_ci * | (__| |_| | _ <| |___ 613498266Sopenharmony_ci * \___|\___/|_| \_\_____| 713498266Sopenharmony_ci * 813498266Sopenharmony_ci * Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al. 913498266Sopenharmony_ci * 1013498266Sopenharmony_ci * This software is licensed as described in the file COPYING, which 1113498266Sopenharmony_ci * you should have received as part of this distribution. The terms 1213498266Sopenharmony_ci * are also available at https://curl.se/docs/copyright.html. 1313498266Sopenharmony_ci * 1413498266Sopenharmony_ci * You may opt to use, copy, modify, merge, publish, distribute and/or sell 1513498266Sopenharmony_ci * copies of the Software, and permit persons to whom the Software is 1613498266Sopenharmony_ci * furnished to do so, under the terms of the COPYING file. 1713498266Sopenharmony_ci * 1813498266Sopenharmony_ci * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY 1913498266Sopenharmony_ci * KIND, either express or implied. 2013498266Sopenharmony_ci * 2113498266Sopenharmony_ci * SPDX-License-Identifier: curl 2213498266Sopenharmony_ci * 2313498266Sopenharmony_ci ***************************************************************************/ 2413498266Sopenharmony_ci/* <DESC> 2513498266Sopenharmony_ci * Websockets data echos 2613498266Sopenharmony_ci * </DESC> 2713498266Sopenharmony_ci */ 2813498266Sopenharmony_ci 2913498266Sopenharmony_ci/* curl stuff */ 3013498266Sopenharmony_ci#include "curl_setup.h" 3113498266Sopenharmony_ci#include <curl/curl.h> 3213498266Sopenharmony_ci 3313498266Sopenharmony_ci#include <stdio.h> 3413498266Sopenharmony_ci#include <stdlib.h> 3513498266Sopenharmony_ci#include <string.h> 3613498266Sopenharmony_ci 3713498266Sopenharmony_ci/* somewhat unix-specific */ 3813498266Sopenharmony_ci#include <sys/time.h> 3913498266Sopenharmony_ci#include <unistd.h> 4013498266Sopenharmony_ci 4113498266Sopenharmony_ci#ifdef USE_WEBSOCKETS 4213498266Sopenharmony_ci 4313498266Sopenharmony_cistatic 4413498266Sopenharmony_civoid dump(const char *text, unsigned char *ptr, size_t size, 4513498266Sopenharmony_ci char nohex) 4613498266Sopenharmony_ci{ 4713498266Sopenharmony_ci size_t i; 4813498266Sopenharmony_ci size_t c; 4913498266Sopenharmony_ci 5013498266Sopenharmony_ci unsigned int width = 0x10; 5113498266Sopenharmony_ci 5213498266Sopenharmony_ci if(nohex) 5313498266Sopenharmony_ci /* without the hex output, we can fit more on screen */ 5413498266Sopenharmony_ci width = 0x40; 5513498266Sopenharmony_ci 5613498266Sopenharmony_ci fprintf(stderr, "%s, %lu bytes (0x%lx)\n", 5713498266Sopenharmony_ci text, (unsigned long)size, (unsigned long)size); 5813498266Sopenharmony_ci 5913498266Sopenharmony_ci for(i = 0; i<size; i += width) { 6013498266Sopenharmony_ci 6113498266Sopenharmony_ci fprintf(stderr, "%4.4lx: ", (unsigned long)i); 6213498266Sopenharmony_ci 6313498266Sopenharmony_ci if(!nohex) { 6413498266Sopenharmony_ci /* hex not disabled, show it */ 6513498266Sopenharmony_ci for(c = 0; c < width; c++) 6613498266Sopenharmony_ci if(i + c < size) 6713498266Sopenharmony_ci fprintf(stderr, "%02x ", ptr[i + c]); 6813498266Sopenharmony_ci else 6913498266Sopenharmony_ci fputs(" ", stderr); 7013498266Sopenharmony_ci } 7113498266Sopenharmony_ci 7213498266Sopenharmony_ci for(c = 0; (c < width) && (i + c < size); c++) { 7313498266Sopenharmony_ci /* check for 0D0A; if found, skip past and start a new line of output */ 7413498266Sopenharmony_ci if(nohex && (i + c + 1 < size) && ptr[i + c] == 0x0D && 7513498266Sopenharmony_ci ptr[i + c + 1] == 0x0A) { 7613498266Sopenharmony_ci i += (c + 2 - width); 7713498266Sopenharmony_ci break; 7813498266Sopenharmony_ci } 7913498266Sopenharmony_ci fprintf(stderr, "%c", 8013498266Sopenharmony_ci (ptr[i + c] >= 0x20) && (ptr[i + c]<0x80)?ptr[i + c]:'.'); 8113498266Sopenharmony_ci /* check again for 0D0A, to avoid an extra \n if it's at width */ 8213498266Sopenharmony_ci if(nohex && (i + c + 2 < size) && ptr[i + c + 1] == 0x0D && 8313498266Sopenharmony_ci ptr[i + c + 2] == 0x0A) { 8413498266Sopenharmony_ci i += (c + 3 - width); 8513498266Sopenharmony_ci break; 8613498266Sopenharmony_ci } 8713498266Sopenharmony_ci } 8813498266Sopenharmony_ci fputc('\n', stderr); /* newline */ 8913498266Sopenharmony_ci } 9013498266Sopenharmony_ci} 9113498266Sopenharmony_ci 9213498266Sopenharmony_cistatic CURLcode send_binary(CURL *curl, char *buf, size_t buflen) 9313498266Sopenharmony_ci{ 9413498266Sopenharmony_ci size_t nwritten; 9513498266Sopenharmony_ci CURLcode result = 9613498266Sopenharmony_ci curl_ws_send(curl, buf, buflen, &nwritten, 0, CURLWS_BINARY); 9713498266Sopenharmony_ci fprintf(stderr, "ws: send_binary(len=%ld) -> %d, %ld\n", 9813498266Sopenharmony_ci (long)buflen, result, (long)nwritten); 9913498266Sopenharmony_ci return result; 10013498266Sopenharmony_ci} 10113498266Sopenharmony_ci 10213498266Sopenharmony_cistatic CURLcode recv_binary(CURL *curl, char *exp_data, size_t exp_len) 10313498266Sopenharmony_ci{ 10413498266Sopenharmony_ci const struct curl_ws_frame *frame; 10513498266Sopenharmony_ci char recvbuf[256]; 10613498266Sopenharmony_ci size_t r_offset, nread; 10713498266Sopenharmony_ci CURLcode result; 10813498266Sopenharmony_ci 10913498266Sopenharmony_ci fprintf(stderr, "recv_binary: expected payload %ld bytes\n", (long)exp_len); 11013498266Sopenharmony_ci r_offset = 0; 11113498266Sopenharmony_ci while(1) { 11213498266Sopenharmony_ci result = curl_ws_recv(curl, recvbuf, sizeof(recvbuf), &nread, &frame); 11313498266Sopenharmony_ci if(result == CURLE_AGAIN) { 11413498266Sopenharmony_ci fprintf(stderr, "EAGAIN, sleep, try again\n"); 11513498266Sopenharmony_ci usleep(100*1000); 11613498266Sopenharmony_ci continue; 11713498266Sopenharmony_ci } 11813498266Sopenharmony_ci fprintf(stderr, "ws: curl_ws_recv(offset=%ld, len=%ld) -> %d, %ld\n", 11913498266Sopenharmony_ci (long)r_offset, (long)sizeof(recvbuf), result, (long)nread); 12013498266Sopenharmony_ci if(result) { 12113498266Sopenharmony_ci return result; 12213498266Sopenharmony_ci } 12313498266Sopenharmony_ci if(!(frame->flags & CURLWS_BINARY)) { 12413498266Sopenharmony_ci fprintf(stderr, "recv_data: wrong frame, got %ld bytes rflags %x\n", 12513498266Sopenharmony_ci (long)nread, frame->flags); 12613498266Sopenharmony_ci return CURLE_RECV_ERROR; 12713498266Sopenharmony_ci } 12813498266Sopenharmony_ci if(frame->offset != (curl_off_t)r_offset) { 12913498266Sopenharmony_ci fprintf(stderr, "recv_data: frame offset, expected %ld, got %ld\n", 13013498266Sopenharmony_ci (long)r_offset, (long)frame->offset); 13113498266Sopenharmony_ci return CURLE_RECV_ERROR; 13213498266Sopenharmony_ci } 13313498266Sopenharmony_ci if(frame->bytesleft != (curl_off_t)(exp_len - r_offset - nread)) { 13413498266Sopenharmony_ci fprintf(stderr, "recv_data: frame bytesleft, expected %ld, got %ld\n", 13513498266Sopenharmony_ci (long)(exp_len - r_offset - nread), (long)frame->bytesleft); 13613498266Sopenharmony_ci return CURLE_RECV_ERROR; 13713498266Sopenharmony_ci } 13813498266Sopenharmony_ci if(r_offset + nread > exp_len) { 13913498266Sopenharmony_ci fprintf(stderr, "recv_data: data length, expected %ld, now at %ld\n", 14013498266Sopenharmony_ci (long)exp_len, (long)(r_offset + nread)); 14113498266Sopenharmony_ci return CURLE_RECV_ERROR; 14213498266Sopenharmony_ci } 14313498266Sopenharmony_ci if(memcmp(exp_data + r_offset, recvbuf, nread)) { 14413498266Sopenharmony_ci fprintf(stderr, "recv_data: data differs, offset=%ld, len=%ld\n", 14513498266Sopenharmony_ci (long)r_offset, (long)nread); 14613498266Sopenharmony_ci dump("expected:", (unsigned char *)exp_data + r_offset, nread, 0); 14713498266Sopenharmony_ci dump("received:", (unsigned char *)recvbuf, nread, 0); 14813498266Sopenharmony_ci return CURLE_RECV_ERROR; 14913498266Sopenharmony_ci } 15013498266Sopenharmony_ci r_offset += nread; 15113498266Sopenharmony_ci if(r_offset >= exp_len) { 15213498266Sopenharmony_ci fprintf(stderr, "recv_data: frame complete\n"); 15313498266Sopenharmony_ci break; 15413498266Sopenharmony_ci } 15513498266Sopenharmony_ci } 15613498266Sopenharmony_ci return CURLE_OK; 15713498266Sopenharmony_ci} 15813498266Sopenharmony_ci 15913498266Sopenharmony_ci/* just close the connection */ 16013498266Sopenharmony_cistatic void websocket_close(CURL *curl) 16113498266Sopenharmony_ci{ 16213498266Sopenharmony_ci size_t sent; 16313498266Sopenharmony_ci CURLcode result = 16413498266Sopenharmony_ci curl_ws_send(curl, "", 0, &sent, 0, CURLWS_CLOSE); 16513498266Sopenharmony_ci fprintf(stderr, 16613498266Sopenharmony_ci "ws: curl_ws_send returned %u, sent %u\n", (int)result, (int)sent); 16713498266Sopenharmony_ci} 16813498266Sopenharmony_ci 16913498266Sopenharmony_cistatic CURLcode data_echo(CURL *curl, size_t plen_min, size_t plen_max) 17013498266Sopenharmony_ci{ 17113498266Sopenharmony_ci CURLcode res; 17213498266Sopenharmony_ci size_t len; 17313498266Sopenharmony_ci char *send_buf; 17413498266Sopenharmony_ci size_t i; 17513498266Sopenharmony_ci 17613498266Sopenharmony_ci send_buf = calloc(1, plen_max); 17713498266Sopenharmony_ci if(!send_buf) 17813498266Sopenharmony_ci return CURLE_OUT_OF_MEMORY; 17913498266Sopenharmony_ci for(i = 0; i < plen_max; ++i) { 18013498266Sopenharmony_ci send_buf[i] = (char)('0' + ((int)i % 10)); 18113498266Sopenharmony_ci } 18213498266Sopenharmony_ci 18313498266Sopenharmony_ci for(len = plen_min; len <= plen_max; ++len) { 18413498266Sopenharmony_ci res = send_binary(curl, send_buf, len); 18513498266Sopenharmony_ci if(res) 18613498266Sopenharmony_ci goto out; 18713498266Sopenharmony_ci res = recv_binary(curl, send_buf, len); 18813498266Sopenharmony_ci if(res) { 18913498266Sopenharmony_ci fprintf(stderr, "recv_data(len=%ld) -> %d\n", (long)len, res); 19013498266Sopenharmony_ci goto out; 19113498266Sopenharmony_ci } 19213498266Sopenharmony_ci } 19313498266Sopenharmony_ci 19413498266Sopenharmony_ciout: 19513498266Sopenharmony_ci if(!res) 19613498266Sopenharmony_ci websocket_close(curl); 19713498266Sopenharmony_ci free(send_buf); 19813498266Sopenharmony_ci return res; 19913498266Sopenharmony_ci} 20013498266Sopenharmony_ci 20113498266Sopenharmony_ci#endif 20213498266Sopenharmony_ci 20313498266Sopenharmony_ciint main(int argc, char *argv[]) 20413498266Sopenharmony_ci{ 20513498266Sopenharmony_ci#ifdef USE_WEBSOCKETS 20613498266Sopenharmony_ci CURL *curl; 20713498266Sopenharmony_ci CURLcode res = CURLE_OK; 20813498266Sopenharmony_ci const char *url; 20913498266Sopenharmony_ci long l1, l2; 21013498266Sopenharmony_ci size_t plen_min, plen_max; 21113498266Sopenharmony_ci 21213498266Sopenharmony_ci 21313498266Sopenharmony_ci if(argc != 4) { 21413498266Sopenharmony_ci fprintf(stderr, "usage: ws-data url minlen maxlen\n"); 21513498266Sopenharmony_ci return 2; 21613498266Sopenharmony_ci } 21713498266Sopenharmony_ci url = argv[1]; 21813498266Sopenharmony_ci l1 = strtol(argv[2], NULL, 10); 21913498266Sopenharmony_ci if(l1 < 0) { 22013498266Sopenharmony_ci fprintf(stderr, "minlen must be >= 0, got %ld\n", l1); 22113498266Sopenharmony_ci return 2; 22213498266Sopenharmony_ci } 22313498266Sopenharmony_ci l2 = strtol(argv[3], NULL, 10); 22413498266Sopenharmony_ci if(l2 < 0) { 22513498266Sopenharmony_ci fprintf(stderr, "maxlen must be >= 0, got %ld\n", l2); 22613498266Sopenharmony_ci return 2; 22713498266Sopenharmony_ci } 22813498266Sopenharmony_ci plen_min = l1; 22913498266Sopenharmony_ci plen_max = l2; 23013498266Sopenharmony_ci if(plen_max < plen_min) { 23113498266Sopenharmony_ci fprintf(stderr, "maxlen must be >= minlen, got %ld-%ld\n", 23213498266Sopenharmony_ci (long)plen_min, (long)plen_max); 23313498266Sopenharmony_ci return 2; 23413498266Sopenharmony_ci } 23513498266Sopenharmony_ci 23613498266Sopenharmony_ci curl_global_init(CURL_GLOBAL_ALL); 23713498266Sopenharmony_ci 23813498266Sopenharmony_ci curl = curl_easy_init(); 23913498266Sopenharmony_ci if(curl) { 24013498266Sopenharmony_ci curl_easy_setopt(curl, CURLOPT_URL, url); 24113498266Sopenharmony_ci 24213498266Sopenharmony_ci /* use the callback style */ 24313498266Sopenharmony_ci curl_easy_setopt(curl, CURLOPT_USERAGENT, "ws-data"); 24413498266Sopenharmony_ci curl_easy_setopt(curl, CURLOPT_VERBOSE, 1L); 24513498266Sopenharmony_ci curl_easy_setopt(curl, CURLOPT_CONNECT_ONLY, 2L); /* websocket style */ 24613498266Sopenharmony_ci res = curl_easy_perform(curl); 24713498266Sopenharmony_ci fprintf(stderr, "curl_easy_perform() returned %u\n", (int)res); 24813498266Sopenharmony_ci if(res == CURLE_OK) 24913498266Sopenharmony_ci res = data_echo(curl, plen_min, plen_max); 25013498266Sopenharmony_ci 25113498266Sopenharmony_ci /* always cleanup */ 25213498266Sopenharmony_ci curl_easy_cleanup(curl); 25313498266Sopenharmony_ci } 25413498266Sopenharmony_ci curl_global_cleanup(); 25513498266Sopenharmony_ci return (int)res; 25613498266Sopenharmony_ci 25713498266Sopenharmony_ci#else /* USE_WEBSOCKETS */ 25813498266Sopenharmony_ci (void)argc; 25913498266Sopenharmony_ci (void)argv; 26013498266Sopenharmony_ci fprintf(stderr, "websockets not enabled in libcurl\n"); 26113498266Sopenharmony_ci return 1; 26213498266Sopenharmony_ci#endif /* !USE_WEBSOCKETS */ 26313498266Sopenharmony_ci} 264