123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334 |
- /***************************************************************************
- * _ _ ____ _
- * Project ___| | | | _ \| |
- * / __| | | | |_) | |
- * | (__| |_| | _ <| |___
- * \___|\___/|_| \_\_____|
- *
- * Copyright (C) 1998 - 2014, 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 http://curl.haxx.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.
- *
- ***************************************************************************/
- #include "test.h"
- #include <curl/mprintf.h>
- #include "memdebug.h"
- static const char *HOSTHEADER = "Host: www.host.foo.com";
- static const char *JAR = "log/jar506";
- #define THREADS 2
- /* struct containing data of a thread */
- struct Tdata {
- CURLSH *share;
- char *url;
- };
- struct userdata {
- char *text;
- int counter;
- };
- int lock[3];
- /* lock callback */
- static void my_lock(CURL *handle, curl_lock_data data, curl_lock_access laccess,
- void *useptr )
- {
- const char *what;
- struct userdata *user = (struct userdata *)useptr;
- int locknum;
- (void)handle;
- (void)laccess;
- switch ( data ) {
- case CURL_LOCK_DATA_SHARE:
- what = "share";
- locknum = 0;
- break;
- case CURL_LOCK_DATA_DNS:
- what = "dns";
- locknum = 1;
- break;
- case CURL_LOCK_DATA_COOKIE:
- what = "cookie";
- locknum = 2;
- break;
- default:
- fprintf(stderr, "lock: no such data: %d\n", (int)data);
- return;
- }
- /* detect locking of locked locks */
- if(lock[locknum]) {
- printf("lock: double locked %s\n", what);
- return;
- }
- lock[locknum]++;
- printf("lock: %-6s [%s]: %d\n", what, user->text, user->counter);
- user->counter++;
- }
- /* unlock callback */
- static void my_unlock(CURL *handle, curl_lock_data data, void *useptr )
- {
- const char *what;
- struct userdata *user = (struct userdata *)useptr;
- int locknum;
- (void)handle;
- switch ( data ) {
- case CURL_LOCK_DATA_SHARE:
- what = "share";
- locknum = 0;
- break;
- case CURL_LOCK_DATA_DNS:
- what = "dns";
- locknum = 1;
- break;
- case CURL_LOCK_DATA_COOKIE:
- what = "cookie";
- locknum = 2;
- break;
- default:
- fprintf(stderr, "unlock: no such data: %d\n", (int)data);
- return;
- }
- /* detect unlocking of unlocked locks */
- if(!lock[locknum]) {
- printf("unlock: double unlocked %s\n", what);
- return;
- }
- lock[locknum]--;
- printf("unlock: %-6s [%s]: %d\n", what, user->text, user->counter);
- user->counter++;
- }
- /* build host entry */
- static struct curl_slist *sethost(struct curl_slist *headers)
- {
- (void)headers;
- return curl_slist_append(NULL, HOSTHEADER );
- }
- /* the dummy thread function */
- static void *fire(void *ptr)
- {
- CURLcode code;
- struct curl_slist *headers;
- struct Tdata *tdata = (struct Tdata*)ptr;
- CURL *curl;
- int i=0;
- if ((curl = curl_easy_init()) == NULL) {
- fprintf(stderr, "curl_easy_init() failed\n");
- return NULL;
- }
- headers = sethost(NULL);
- curl_easy_setopt(curl, CURLOPT_VERBOSE, 1L);
- curl_easy_setopt(curl, CURLOPT_HTTPHEADER, headers);
- curl_easy_setopt(curl, CURLOPT_URL, tdata->url);
- printf( "CURLOPT_SHARE\n" );
- curl_easy_setopt(curl, CURLOPT_SHARE, tdata->share);
- printf( "PERFORM\n" );
- code = curl_easy_perform(curl);
- if( code != CURLE_OK ) {
- fprintf(stderr, "perform url '%s' repeat %d failed, curlcode %d\n",
- tdata->url, i, (int)code);
- }
- printf( "CLEANUP\n" );
- curl_easy_cleanup(curl);
- curl_slist_free_all(headers);
- return NULL;
- }
- /* build request url */
- static char *suburl(const char *base, int i)
- {
- return curl_maprintf("%s%.4d", base, i);
- }
- /* test function */
- int test(char *URL)
- {
- int res;
- CURLSHcode scode = CURLSHE_OK;
- char *url = NULL;
- struct Tdata tdata;
- CURL *curl;
- CURLSH *share;
- struct curl_slist *headers = NULL;
- int i;
- struct userdata user;
- user.text = (char *)"Pigs in space";
- user.counter = 0;
- printf( "GLOBAL_INIT\n" );
- if (curl_global_init(CURL_GLOBAL_ALL) != CURLE_OK) {
- fprintf(stderr, "curl_global_init() failed\n");
- return TEST_ERR_MAJOR_BAD;
- }
- /* prepare share */
- printf( "SHARE_INIT\n" );
- if ((share = curl_share_init()) == NULL) {
- fprintf(stderr, "curl_share_init() failed\n");
- curl_global_cleanup();
- return TEST_ERR_MAJOR_BAD;
- }
- if ( CURLSHE_OK == scode ) {
- printf( "CURLSHOPT_LOCKFUNC\n" );
- scode = curl_share_setopt( share, CURLSHOPT_LOCKFUNC, my_lock);
- }
- if ( CURLSHE_OK == scode ) {
- printf( "CURLSHOPT_UNLOCKFUNC\n" );
- scode = curl_share_setopt( share, CURLSHOPT_UNLOCKFUNC, my_unlock);
- }
- if ( CURLSHE_OK == scode ) {
- printf( "CURLSHOPT_USERDATA\n" );
- scode = curl_share_setopt( share, CURLSHOPT_USERDATA, &user);
- }
- if ( CURLSHE_OK == scode ) {
- printf( "CURL_LOCK_DATA_COOKIE\n" );
- scode = curl_share_setopt( share, CURLSHOPT_SHARE, CURL_LOCK_DATA_COOKIE);
- }
- if ( CURLSHE_OK == scode ) {
- printf( "CURL_LOCK_DATA_DNS\n" );
- scode = curl_share_setopt( share, CURLSHOPT_SHARE, CURL_LOCK_DATA_DNS);
- }
- if ( CURLSHE_OK != scode ) {
- fprintf(stderr, "curl_share_setopt() failed\n");
- curl_share_cleanup(share);
- curl_global_cleanup();
- return TEST_ERR_MAJOR_BAD;
- }
- /* initial cookie manipulation */
- if ((curl = curl_easy_init()) == NULL) {
- fprintf(stderr, "curl_easy_init() failed\n");
- curl_share_cleanup(share);
- curl_global_cleanup();
- return TEST_ERR_MAJOR_BAD;
- }
- printf( "CURLOPT_SHARE\n" );
- test_setopt( curl, CURLOPT_SHARE, share );
- printf( "CURLOPT_COOKIELIST injected_and_clobbered\n" );
- test_setopt( curl, CURLOPT_COOKIELIST,
- "Set-Cookie: injected_and_clobbered=yes; "
- "domain=host.foo.com; expires=Sat Feb 2 11:56:27 GMT 2030" );
- printf( "CURLOPT_COOKIELIST ALL\n" );
- test_setopt( curl, CURLOPT_COOKIELIST, "ALL" );
- printf( "CURLOPT_COOKIELIST session\n" );
- test_setopt( curl, CURLOPT_COOKIELIST, "Set-Cookie: session=elephants" );
- printf( "CURLOPT_COOKIELIST injected\n" );
- test_setopt( curl, CURLOPT_COOKIELIST,
- "Set-Cookie: injected=yes; domain=host.foo.com; "
- "expires=Sat Feb 2 11:56:27 GMT 2030" );
- printf( "CURLOPT_COOKIELIST SESS\n" );
- test_setopt( curl, CURLOPT_COOKIELIST, "SESS" );
- printf( "CLEANUP\n" );
- curl_easy_cleanup( curl );
- res = 0;
- /* start treads */
- for (i=1; i<=THREADS; i++ ) {
- /* set thread data */
- tdata.url = suburl( URL, i ); /* must be curl_free()d */
- tdata.share = share;
- /* simulate thread, direct call of "thread" function */
- printf( "*** run %d\n",i );
- fire( &tdata );
- curl_free( tdata.url );
- }
- /* fetch a another one and save cookies */
- printf( "*** run %d\n", i );
- if ((curl = curl_easy_init()) == NULL) {
- fprintf(stderr, "curl_easy_init() failed\n");
- curl_share_cleanup(share);
- curl_global_cleanup();
- return TEST_ERR_MAJOR_BAD;
- }
- url = suburl( URL, i );
- headers = sethost( NULL );
- test_setopt( curl, CURLOPT_HTTPHEADER, headers );
- test_setopt( curl, CURLOPT_URL, url );
- printf( "CURLOPT_SHARE\n" );
- test_setopt( curl, CURLOPT_SHARE, share );
- printf( "CURLOPT_COOKIEJAR\n" );
- test_setopt( curl, CURLOPT_COOKIEJAR, JAR );
- printf( "CURLOPT_COOKIELIST FLUSH\n" );
- test_setopt( curl, CURLOPT_COOKIELIST, "FLUSH" );
- printf( "PERFORM\n" );
- curl_easy_perform( curl );
- /* try to free share, expect to fail because share is in use*/
- printf( "try SHARE_CLEANUP...\n" );
- scode = curl_share_cleanup( share );
- if ( scode==CURLSHE_OK )
- {
- fprintf(stderr, "curl_share_cleanup succeed but error expected\n");
- share = NULL;
- } else {
- printf( "SHARE_CLEANUP failed, correct\n" );
- }
- test_cleanup:
- /* clean up last handle */
- printf( "CLEANUP\n" );
- curl_easy_cleanup( curl );
- if ( headers )
- curl_slist_free_all( headers );
- if ( url )
- curl_free(url);
- /* free share */
- printf( "SHARE_CLEANUP\n" );
- scode = curl_share_cleanup( share );
- if ( scode!=CURLSHE_OK )
- fprintf(stderr, "curl_share_cleanup failed, code errno %d\n",
- (int)scode);
- printf( "GLOBAL_CLEANUP\n" );
- curl_global_cleanup();
- return res;
- }
|