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#include "test.h" 2513498266Sopenharmony_ci#include "memdebug.h" 2613498266Sopenharmony_ci 2713498266Sopenharmony_cistatic const char * const HOSTHEADER = "Host: www.host.foo.com"; 2813498266Sopenharmony_ci#define JAR libtest_arg2 2913498266Sopenharmony_ci#define THREADS 2 3013498266Sopenharmony_ci 3113498266Sopenharmony_ci/* struct containing data of a thread */ 3213498266Sopenharmony_cistruct Tdata { 3313498266Sopenharmony_ci CURLSH *share; 3413498266Sopenharmony_ci char *url; 3513498266Sopenharmony_ci}; 3613498266Sopenharmony_ci 3713498266Sopenharmony_cistruct userdata { 3813498266Sopenharmony_ci const char *text; 3913498266Sopenharmony_ci int counter; 4013498266Sopenharmony_ci}; 4113498266Sopenharmony_ci 4213498266Sopenharmony_cistatic int locks[3]; 4313498266Sopenharmony_ci 4413498266Sopenharmony_ci/* lock callback */ 4513498266Sopenharmony_cistatic void my_lock(CURL *handle, curl_lock_data data, 4613498266Sopenharmony_ci curl_lock_access laccess, void *useptr) 4713498266Sopenharmony_ci{ 4813498266Sopenharmony_ci const char *what; 4913498266Sopenharmony_ci struct userdata *user = (struct userdata *)useptr; 5013498266Sopenharmony_ci int locknum; 5113498266Sopenharmony_ci 5213498266Sopenharmony_ci (void)handle; 5313498266Sopenharmony_ci (void)laccess; 5413498266Sopenharmony_ci 5513498266Sopenharmony_ci switch(data) { 5613498266Sopenharmony_ci case CURL_LOCK_DATA_SHARE: 5713498266Sopenharmony_ci what = "share"; 5813498266Sopenharmony_ci locknum = 0; 5913498266Sopenharmony_ci break; 6013498266Sopenharmony_ci case CURL_LOCK_DATA_DNS: 6113498266Sopenharmony_ci what = "dns"; 6213498266Sopenharmony_ci locknum = 1; 6313498266Sopenharmony_ci break; 6413498266Sopenharmony_ci case CURL_LOCK_DATA_COOKIE: 6513498266Sopenharmony_ci what = "cookie"; 6613498266Sopenharmony_ci locknum = 2; 6713498266Sopenharmony_ci break; 6813498266Sopenharmony_ci default: 6913498266Sopenharmony_ci fprintf(stderr, "lock: no such data: %d\n", (int)data); 7013498266Sopenharmony_ci return; 7113498266Sopenharmony_ci } 7213498266Sopenharmony_ci 7313498266Sopenharmony_ci /* detect locking of locked locks */ 7413498266Sopenharmony_ci if(locks[locknum]) { 7513498266Sopenharmony_ci printf("lock: double locked %s\n", what); 7613498266Sopenharmony_ci return; 7713498266Sopenharmony_ci } 7813498266Sopenharmony_ci locks[locknum]++; 7913498266Sopenharmony_ci 8013498266Sopenharmony_ci printf("lock: %-6s [%s]: %d\n", what, user->text, user->counter); 8113498266Sopenharmony_ci user->counter++; 8213498266Sopenharmony_ci} 8313498266Sopenharmony_ci 8413498266Sopenharmony_ci/* unlock callback */ 8513498266Sopenharmony_cistatic void my_unlock(CURL *handle, curl_lock_data data, void *useptr) 8613498266Sopenharmony_ci{ 8713498266Sopenharmony_ci const char *what; 8813498266Sopenharmony_ci struct userdata *user = (struct userdata *)useptr; 8913498266Sopenharmony_ci int locknum; 9013498266Sopenharmony_ci (void)handle; 9113498266Sopenharmony_ci switch(data) { 9213498266Sopenharmony_ci case CURL_LOCK_DATA_SHARE: 9313498266Sopenharmony_ci what = "share"; 9413498266Sopenharmony_ci locknum = 0; 9513498266Sopenharmony_ci break; 9613498266Sopenharmony_ci case CURL_LOCK_DATA_DNS: 9713498266Sopenharmony_ci what = "dns"; 9813498266Sopenharmony_ci locknum = 1; 9913498266Sopenharmony_ci break; 10013498266Sopenharmony_ci case CURL_LOCK_DATA_COOKIE: 10113498266Sopenharmony_ci what = "cookie"; 10213498266Sopenharmony_ci locknum = 2; 10313498266Sopenharmony_ci break; 10413498266Sopenharmony_ci default: 10513498266Sopenharmony_ci fprintf(stderr, "unlock: no such data: %d\n", (int)data); 10613498266Sopenharmony_ci return; 10713498266Sopenharmony_ci } 10813498266Sopenharmony_ci 10913498266Sopenharmony_ci /* detect unlocking of unlocked locks */ 11013498266Sopenharmony_ci if(!locks[locknum]) { 11113498266Sopenharmony_ci printf("unlock: double unlocked %s\n", what); 11213498266Sopenharmony_ci return; 11313498266Sopenharmony_ci } 11413498266Sopenharmony_ci locks[locknum]--; 11513498266Sopenharmony_ci 11613498266Sopenharmony_ci printf("unlock: %-6s [%s]: %d\n", what, user->text, user->counter); 11713498266Sopenharmony_ci user->counter++; 11813498266Sopenharmony_ci} 11913498266Sopenharmony_ci 12013498266Sopenharmony_ci 12113498266Sopenharmony_ci/* build host entry */ 12213498266Sopenharmony_cistatic struct curl_slist *sethost(struct curl_slist *headers) 12313498266Sopenharmony_ci{ 12413498266Sopenharmony_ci (void)headers; 12513498266Sopenharmony_ci return curl_slist_append(NULL, HOSTHEADER); 12613498266Sopenharmony_ci} 12713498266Sopenharmony_ci 12813498266Sopenharmony_ci 12913498266Sopenharmony_ci/* the dummy thread function */ 13013498266Sopenharmony_cistatic void *fire(void *ptr) 13113498266Sopenharmony_ci{ 13213498266Sopenharmony_ci CURLcode code; 13313498266Sopenharmony_ci struct curl_slist *headers; 13413498266Sopenharmony_ci struct Tdata *tdata = (struct Tdata*)ptr; 13513498266Sopenharmony_ci CURL *curl; 13613498266Sopenharmony_ci 13713498266Sopenharmony_ci curl = curl_easy_init(); 13813498266Sopenharmony_ci if(!curl) { 13913498266Sopenharmony_ci fprintf(stderr, "curl_easy_init() failed\n"); 14013498266Sopenharmony_ci return NULL; 14113498266Sopenharmony_ci } 14213498266Sopenharmony_ci 14313498266Sopenharmony_ci headers = sethost(NULL); 14413498266Sopenharmony_ci curl_easy_setopt(curl, CURLOPT_VERBOSE, 1L); 14513498266Sopenharmony_ci curl_easy_setopt(curl, CURLOPT_HTTPHEADER, headers); 14613498266Sopenharmony_ci curl_easy_setopt(curl, CURLOPT_URL, tdata->url); 14713498266Sopenharmony_ci curl_easy_setopt(curl, CURLOPT_COOKIEFILE, ""); 14813498266Sopenharmony_ci printf("CURLOPT_SHARE\n"); 14913498266Sopenharmony_ci curl_easy_setopt(curl, CURLOPT_SHARE, tdata->share); 15013498266Sopenharmony_ci 15113498266Sopenharmony_ci printf("PERFORM\n"); 15213498266Sopenharmony_ci code = curl_easy_perform(curl); 15313498266Sopenharmony_ci if(code) { 15413498266Sopenharmony_ci int i = 0; 15513498266Sopenharmony_ci fprintf(stderr, "perform url '%s' repeat %d failed, curlcode %d\n", 15613498266Sopenharmony_ci tdata->url, i, (int)code); 15713498266Sopenharmony_ci } 15813498266Sopenharmony_ci 15913498266Sopenharmony_ci printf("CLEANUP\n"); 16013498266Sopenharmony_ci curl_easy_cleanup(curl); 16113498266Sopenharmony_ci curl_slist_free_all(headers); 16213498266Sopenharmony_ci 16313498266Sopenharmony_ci return NULL; 16413498266Sopenharmony_ci} 16513498266Sopenharmony_ci 16613498266Sopenharmony_ci 16713498266Sopenharmony_ci/* build request url */ 16813498266Sopenharmony_cistatic char *suburl(const char *base, int i) 16913498266Sopenharmony_ci{ 17013498266Sopenharmony_ci return curl_maprintf("%s%.4d", base, i); 17113498266Sopenharmony_ci} 17213498266Sopenharmony_ci 17313498266Sopenharmony_ci 17413498266Sopenharmony_ci/* test function */ 17513498266Sopenharmony_ciint test(char *URL) 17613498266Sopenharmony_ci{ 17713498266Sopenharmony_ci int res; 17813498266Sopenharmony_ci CURLSHcode scode = CURLSHE_OK; 17913498266Sopenharmony_ci CURLcode code = CURLE_OK; 18013498266Sopenharmony_ci char *url = NULL; 18113498266Sopenharmony_ci struct Tdata tdata; 18213498266Sopenharmony_ci CURL *curl; 18313498266Sopenharmony_ci CURLSH *share; 18413498266Sopenharmony_ci struct curl_slist *headers = NULL; 18513498266Sopenharmony_ci struct curl_slist *cookies = NULL; 18613498266Sopenharmony_ci struct curl_slist *next_cookie = NULL; 18713498266Sopenharmony_ci int i; 18813498266Sopenharmony_ci struct userdata user; 18913498266Sopenharmony_ci 19013498266Sopenharmony_ci user.text = "Pigs in space"; 19113498266Sopenharmony_ci user.counter = 0; 19213498266Sopenharmony_ci 19313498266Sopenharmony_ci printf("GLOBAL_INIT\n"); 19413498266Sopenharmony_ci if(curl_global_init(CURL_GLOBAL_ALL) != CURLE_OK) { 19513498266Sopenharmony_ci fprintf(stderr, "curl_global_init() failed\n"); 19613498266Sopenharmony_ci return TEST_ERR_MAJOR_BAD; 19713498266Sopenharmony_ci } 19813498266Sopenharmony_ci 19913498266Sopenharmony_ci /* prepare share */ 20013498266Sopenharmony_ci printf("SHARE_INIT\n"); 20113498266Sopenharmony_ci share = curl_share_init(); 20213498266Sopenharmony_ci if(!share) { 20313498266Sopenharmony_ci fprintf(stderr, "curl_share_init() failed\n"); 20413498266Sopenharmony_ci curl_global_cleanup(); 20513498266Sopenharmony_ci return TEST_ERR_MAJOR_BAD; 20613498266Sopenharmony_ci } 20713498266Sopenharmony_ci 20813498266Sopenharmony_ci if(CURLSHE_OK == scode) { 20913498266Sopenharmony_ci printf("CURLSHOPT_LOCKFUNC\n"); 21013498266Sopenharmony_ci scode = curl_share_setopt(share, CURLSHOPT_LOCKFUNC, my_lock); 21113498266Sopenharmony_ci } 21213498266Sopenharmony_ci if(CURLSHE_OK == scode) { 21313498266Sopenharmony_ci printf("CURLSHOPT_UNLOCKFUNC\n"); 21413498266Sopenharmony_ci scode = curl_share_setopt(share, CURLSHOPT_UNLOCKFUNC, my_unlock); 21513498266Sopenharmony_ci } 21613498266Sopenharmony_ci if(CURLSHE_OK == scode) { 21713498266Sopenharmony_ci printf("CURLSHOPT_USERDATA\n"); 21813498266Sopenharmony_ci scode = curl_share_setopt(share, CURLSHOPT_USERDATA, &user); 21913498266Sopenharmony_ci } 22013498266Sopenharmony_ci if(CURLSHE_OK == scode) { 22113498266Sopenharmony_ci printf("CURL_LOCK_DATA_COOKIE\n"); 22213498266Sopenharmony_ci scode = curl_share_setopt(share, CURLSHOPT_SHARE, CURL_LOCK_DATA_COOKIE); 22313498266Sopenharmony_ci } 22413498266Sopenharmony_ci if(CURLSHE_OK == scode) { 22513498266Sopenharmony_ci printf("CURL_LOCK_DATA_DNS\n"); 22613498266Sopenharmony_ci scode = curl_share_setopt(share, CURLSHOPT_SHARE, CURL_LOCK_DATA_DNS); 22713498266Sopenharmony_ci } 22813498266Sopenharmony_ci 22913498266Sopenharmony_ci if(CURLSHE_OK != scode) { 23013498266Sopenharmony_ci fprintf(stderr, "curl_share_setopt() failed\n"); 23113498266Sopenharmony_ci curl_share_cleanup(share); 23213498266Sopenharmony_ci curl_global_cleanup(); 23313498266Sopenharmony_ci return TEST_ERR_MAJOR_BAD; 23413498266Sopenharmony_ci } 23513498266Sopenharmony_ci 23613498266Sopenharmony_ci /* initial cookie manipulation */ 23713498266Sopenharmony_ci curl = curl_easy_init(); 23813498266Sopenharmony_ci if(!curl) { 23913498266Sopenharmony_ci fprintf(stderr, "curl_easy_init() failed\n"); 24013498266Sopenharmony_ci curl_share_cleanup(share); 24113498266Sopenharmony_ci curl_global_cleanup(); 24213498266Sopenharmony_ci return TEST_ERR_MAJOR_BAD; 24313498266Sopenharmony_ci } 24413498266Sopenharmony_ci printf("CURLOPT_SHARE\n"); 24513498266Sopenharmony_ci test_setopt(curl, CURLOPT_SHARE, share); 24613498266Sopenharmony_ci printf("CURLOPT_COOKIELIST injected_and_clobbered\n"); 24713498266Sopenharmony_ci test_setopt(curl, CURLOPT_COOKIELIST, 24813498266Sopenharmony_ci "Set-Cookie: injected_and_clobbered=yes; " 24913498266Sopenharmony_ci "domain=host.foo.com; expires=Sat Feb 2 11:56:27 GMT 2030"); 25013498266Sopenharmony_ci printf("CURLOPT_COOKIELIST ALL\n"); 25113498266Sopenharmony_ci test_setopt(curl, CURLOPT_COOKIELIST, "ALL"); 25213498266Sopenharmony_ci printf("CURLOPT_COOKIELIST session\n"); 25313498266Sopenharmony_ci test_setopt(curl, CURLOPT_COOKIELIST, "Set-Cookie: session=elephants"); 25413498266Sopenharmony_ci printf("CURLOPT_COOKIELIST injected\n"); 25513498266Sopenharmony_ci test_setopt(curl, CURLOPT_COOKIELIST, 25613498266Sopenharmony_ci "Set-Cookie: injected=yes; domain=host.foo.com; " 25713498266Sopenharmony_ci "expires=Sat Feb 2 11:56:27 GMT 2030"); 25813498266Sopenharmony_ci printf("CURLOPT_COOKIELIST SESS\n"); 25913498266Sopenharmony_ci test_setopt(curl, CURLOPT_COOKIELIST, "SESS"); 26013498266Sopenharmony_ci printf("CLEANUP\n"); 26113498266Sopenharmony_ci curl_easy_cleanup(curl); 26213498266Sopenharmony_ci 26313498266Sopenharmony_ci 26413498266Sopenharmony_ci res = 0; 26513498266Sopenharmony_ci 26613498266Sopenharmony_ci /* start treads */ 26713498266Sopenharmony_ci for(i = 1; i <= THREADS; i++) { 26813498266Sopenharmony_ci 26913498266Sopenharmony_ci /* set thread data */ 27013498266Sopenharmony_ci tdata.url = suburl(URL, i); /* must be curl_free()d */ 27113498266Sopenharmony_ci tdata.share = share; 27213498266Sopenharmony_ci 27313498266Sopenharmony_ci /* simulate thread, direct call of "thread" function */ 27413498266Sopenharmony_ci printf("*** run %d\n",i); 27513498266Sopenharmony_ci fire(&tdata); 27613498266Sopenharmony_ci 27713498266Sopenharmony_ci curl_free(tdata.url); 27813498266Sopenharmony_ci } 27913498266Sopenharmony_ci 28013498266Sopenharmony_ci 28113498266Sopenharmony_ci /* fetch a another one and save cookies */ 28213498266Sopenharmony_ci printf("*** run %d\n", i); 28313498266Sopenharmony_ci curl = curl_easy_init(); 28413498266Sopenharmony_ci if(!curl) { 28513498266Sopenharmony_ci fprintf(stderr, "curl_easy_init() failed\n"); 28613498266Sopenharmony_ci curl_share_cleanup(share); 28713498266Sopenharmony_ci curl_global_cleanup(); 28813498266Sopenharmony_ci return TEST_ERR_MAJOR_BAD; 28913498266Sopenharmony_ci } 29013498266Sopenharmony_ci 29113498266Sopenharmony_ci url = suburl(URL, i); 29213498266Sopenharmony_ci headers = sethost(NULL); 29313498266Sopenharmony_ci test_setopt(curl, CURLOPT_HTTPHEADER, headers); 29413498266Sopenharmony_ci test_setopt(curl, CURLOPT_URL, url); 29513498266Sopenharmony_ci printf("CURLOPT_SHARE\n"); 29613498266Sopenharmony_ci test_setopt(curl, CURLOPT_SHARE, share); 29713498266Sopenharmony_ci printf("CURLOPT_COOKIEJAR\n"); 29813498266Sopenharmony_ci test_setopt(curl, CURLOPT_COOKIEJAR, JAR); 29913498266Sopenharmony_ci printf("CURLOPT_COOKIELIST FLUSH\n"); 30013498266Sopenharmony_ci test_setopt(curl, CURLOPT_COOKIELIST, "FLUSH"); 30113498266Sopenharmony_ci 30213498266Sopenharmony_ci printf("PERFORM\n"); 30313498266Sopenharmony_ci curl_easy_perform(curl); 30413498266Sopenharmony_ci 30513498266Sopenharmony_ci printf("CLEANUP\n"); 30613498266Sopenharmony_ci curl_easy_cleanup(curl); 30713498266Sopenharmony_ci curl_free(url); 30813498266Sopenharmony_ci curl_slist_free_all(headers); 30913498266Sopenharmony_ci 31013498266Sopenharmony_ci /* load cookies */ 31113498266Sopenharmony_ci curl = curl_easy_init(); 31213498266Sopenharmony_ci if(!curl) { 31313498266Sopenharmony_ci fprintf(stderr, "curl_easy_init() failed\n"); 31413498266Sopenharmony_ci curl_share_cleanup(share); 31513498266Sopenharmony_ci curl_global_cleanup(); 31613498266Sopenharmony_ci return TEST_ERR_MAJOR_BAD; 31713498266Sopenharmony_ci } 31813498266Sopenharmony_ci url = suburl(URL, i); 31913498266Sopenharmony_ci headers = sethost(NULL); 32013498266Sopenharmony_ci test_setopt(curl, CURLOPT_HTTPHEADER, headers); 32113498266Sopenharmony_ci test_setopt(curl, CURLOPT_URL, url); 32213498266Sopenharmony_ci printf("CURLOPT_SHARE\n"); 32313498266Sopenharmony_ci test_setopt(curl, CURLOPT_SHARE, share); 32413498266Sopenharmony_ci printf("CURLOPT_COOKIELIST ALL\n"); 32513498266Sopenharmony_ci test_setopt(curl, CURLOPT_COOKIELIST, "ALL"); 32613498266Sopenharmony_ci printf("CURLOPT_COOKIEJAR\n"); 32713498266Sopenharmony_ci test_setopt(curl, CURLOPT_COOKIEFILE, JAR); 32813498266Sopenharmony_ci printf("CURLOPT_COOKIELIST RELOAD\n"); 32913498266Sopenharmony_ci test_setopt(curl, CURLOPT_COOKIELIST, "RELOAD"); 33013498266Sopenharmony_ci 33113498266Sopenharmony_ci code = curl_easy_getinfo(curl, CURLINFO_COOKIELIST, &cookies); 33213498266Sopenharmony_ci if(code != CURLE_OK) { 33313498266Sopenharmony_ci fprintf(stderr, "curl_easy_getinfo() failed\n"); 33413498266Sopenharmony_ci res = TEST_ERR_MAJOR_BAD; 33513498266Sopenharmony_ci goto test_cleanup; 33613498266Sopenharmony_ci } 33713498266Sopenharmony_ci printf("loaded cookies:\n"); 33813498266Sopenharmony_ci if(!cookies) { 33913498266Sopenharmony_ci fprintf(stderr, " reloading cookies from '%s' failed\n", JAR); 34013498266Sopenharmony_ci res = TEST_ERR_MAJOR_BAD; 34113498266Sopenharmony_ci goto test_cleanup; 34213498266Sopenharmony_ci } 34313498266Sopenharmony_ci printf("-----------------\n"); 34413498266Sopenharmony_ci next_cookie = cookies; 34513498266Sopenharmony_ci while(next_cookie) { 34613498266Sopenharmony_ci printf(" %s\n", next_cookie->data); 34713498266Sopenharmony_ci next_cookie = next_cookie->next; 34813498266Sopenharmony_ci } 34913498266Sopenharmony_ci printf("-----------------\n"); 35013498266Sopenharmony_ci curl_slist_free_all(cookies); 35113498266Sopenharmony_ci 35213498266Sopenharmony_ci /* try to free share, expect to fail because share is in use */ 35313498266Sopenharmony_ci printf("try SHARE_CLEANUP...\n"); 35413498266Sopenharmony_ci scode = curl_share_cleanup(share); 35513498266Sopenharmony_ci if(scode == CURLSHE_OK) { 35613498266Sopenharmony_ci fprintf(stderr, "curl_share_cleanup succeed but error expected\n"); 35713498266Sopenharmony_ci share = NULL; 35813498266Sopenharmony_ci } 35913498266Sopenharmony_ci else { 36013498266Sopenharmony_ci printf("SHARE_CLEANUP failed, correct\n"); 36113498266Sopenharmony_ci } 36213498266Sopenharmony_ci 36313498266Sopenharmony_citest_cleanup: 36413498266Sopenharmony_ci 36513498266Sopenharmony_ci /* clean up last handle */ 36613498266Sopenharmony_ci printf("CLEANUP\n"); 36713498266Sopenharmony_ci curl_easy_cleanup(curl); 36813498266Sopenharmony_ci curl_slist_free_all(headers); 36913498266Sopenharmony_ci curl_free(url); 37013498266Sopenharmony_ci 37113498266Sopenharmony_ci /* free share */ 37213498266Sopenharmony_ci printf("SHARE_CLEANUP\n"); 37313498266Sopenharmony_ci scode = curl_share_cleanup(share); 37413498266Sopenharmony_ci if(scode != CURLSHE_OK) 37513498266Sopenharmony_ci fprintf(stderr, "curl_share_cleanup failed, code errno %d\n", 37613498266Sopenharmony_ci (int)scode); 37713498266Sopenharmony_ci 37813498266Sopenharmony_ci printf("GLOBAL_CLEANUP\n"); 37913498266Sopenharmony_ci curl_global_cleanup(); 38013498266Sopenharmony_ci 38113498266Sopenharmony_ci return res; 38213498266Sopenharmony_ci} 383