1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636 |
- /* Copyright (c) 2015-2019, The Tor Project, Inc. */
- /* See LICENSE for licensing information */
- #define CONTROL_PRIVATE
- #include "core/or/or.h"
- #include "lib/crypt_ops/crypto_ed25519.h"
- #include "feature/client/bridges.h"
- #include "feature/control/control.h"
- #include "feature/client/entrynodes.h"
- #include "feature/hs/hs_common.h"
- #include "feature/nodelist/networkstatus.h"
- #include "feature/rend/rendservice.h"
- #include "feature/nodelist/authcert.h"
- #include "feature/nodelist/nodelist.h"
- #include "test/test.h"
- #include "test/test_helpers.h"
- #include "lib/net/resolve.h"
- #include "feature/control/control_connection_st.h"
- #include "feature/dirclient/download_status_st.h"
- #include "feature/nodelist/microdesc_st.h"
- #include "feature/nodelist/node_st.h"
- static void
- test_add_onion_helper_keyarg_v3(void *arg)
- {
- int ret, hs_version;
- add_onion_secret_key_t pk;
- char *key_new_blob = NULL;
- char *err_msg = NULL;
- const char *key_new_alg = NULL;
- (void) arg;
- memset(&pk, 0, sizeof(pk));
- /* Test explicit ED25519-V3 key generation. */
- ret = add_onion_helper_keyarg("NEW:ED25519-V3", 0, &key_new_alg,
- &key_new_blob, &pk, &hs_version,
- &err_msg);
- tt_int_op(ret, OP_EQ, 0);
- tt_int_op(hs_version, OP_EQ, HS_VERSION_THREE);
- tt_assert(pk.v3);
- tt_str_op(key_new_alg, OP_EQ, "ED25519-V3");
- tt_assert(key_new_blob);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- tor_free(pk.v3); pk.v3 = NULL;
- tor_free(key_new_blob);
- /* Test discarding the private key. */
- ret = add_onion_helper_keyarg("NEW:ED25519-V3", 1, &key_new_alg,
- &key_new_blob, &pk, &hs_version,
- &err_msg);
- tt_int_op(ret, OP_EQ, 0);
- tt_int_op(hs_version, OP_EQ, HS_VERSION_THREE);
- tt_assert(pk.v3);
- tt_ptr_op(key_new_alg, OP_EQ, NULL);
- tt_ptr_op(key_new_blob, OP_EQ, NULL);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- tor_free(pk.v3); pk.v3 = NULL;
- tor_free(key_new_blob);
- /* Test passing a key blob. */
- {
- /* The base64 key and hex key are the same. Hex key is 64 bytes long. The
- * sk has been generated randomly using python3. */
- const char *base64_sk =
- "a9bT19PqGC9Y+BmOo1IQvCGjjwxMiaaxEXZ+FKMxpEQW"
- "6AmSV5roThUGMRCaqQSCnR2jI1vL2QxHORzI4RxMmw==";
- const char *hex_sk =
- "\x6b\xd6\xd3\xd7\xd3\xea\x18\x2f\x58\xf8\x19\x8e\xa3\x52\x10\xbc"
- "\x21\xa3\x8f\x0c\x4c\x89\xa6\xb1\x11\x76\x7e\x14\xa3\x31\xa4\x44"
- "\x16\xe8\x09\x92\x57\x9a\xe8\x4e\x15\x06\x31\x10\x9a\xa9\x04\x82"
- "\x9d\x1d\xa3\x23\x5b\xcb\xd9\x0c\x47\x39\x1c\xc8\xe1\x1c\x4c\x9b";
- char *key_blob = NULL;
- tor_asprintf(&key_blob, "ED25519-V3:%s", base64_sk);
- tt_assert(key_blob);
- ret = add_onion_helper_keyarg(key_blob, 1, &key_new_alg,
- &key_new_blob, &pk, &hs_version,
- &err_msg);
- tor_free(key_blob);
- tt_int_op(ret, OP_EQ, 0);
- tt_int_op(hs_version, OP_EQ, HS_VERSION_THREE);
- tt_assert(pk.v3);
- tt_mem_op(pk.v3, OP_EQ, hex_sk, 64);
- tt_ptr_op(key_new_alg, OP_EQ, NULL);
- tt_ptr_op(key_new_blob, OP_EQ, NULL);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- tor_free(pk.v3); pk.v3 = NULL;
- tor_free(key_new_blob);
- }
- done:
- tor_free(pk.v3);
- tor_free(key_new_blob);
- tor_free(err_msg);
- }
- static void
- test_add_onion_helper_keyarg_v2(void *arg)
- {
- int ret, hs_version;
- add_onion_secret_key_t pk;
- crypto_pk_t *pk1 = NULL;
- const char *key_new_alg = NULL;
- char *key_new_blob = NULL;
- char *err_msg = NULL;
- char *encoded = NULL;
- char *arg_str = NULL;
- (void) arg;
- memset(&pk, 0, sizeof(pk));
- /* Test explicit RSA1024 key generation. */
- ret = add_onion_helper_keyarg("NEW:RSA1024", 0, &key_new_alg, &key_new_blob,
- &pk, &hs_version, &err_msg);
- tt_int_op(ret, OP_EQ, 0);
- tt_int_op(hs_version, OP_EQ, HS_VERSION_TWO);
- tt_assert(pk.v2);
- tt_str_op(key_new_alg, OP_EQ, "RSA1024");
- tt_assert(key_new_blob);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- /* Test "BEST" key generation (Assumes BEST = RSA1024). */
- crypto_pk_free(pk.v2); pk.v2 = NULL;
- tor_free(key_new_blob);
- ret = add_onion_helper_keyarg("NEW:BEST", 0, &key_new_alg, &key_new_blob,
- &pk, &hs_version, &err_msg);
- tt_int_op(ret, OP_EQ, 0);
- tt_int_op(hs_version, OP_EQ, HS_VERSION_TWO);
- tt_assert(pk.v2);
- tt_str_op(key_new_alg, OP_EQ, "RSA1024");
- tt_assert(key_new_blob);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- /* Test discarding the private key. */
- crypto_pk_free(pk.v2); pk.v2 = NULL;
- tor_free(key_new_blob);
- ret = add_onion_helper_keyarg("NEW:BEST", 1, &key_new_alg, &key_new_blob,
- &pk, &hs_version, &err_msg);
- tt_int_op(ret, OP_EQ, 0);
- tt_int_op(hs_version, OP_EQ, HS_VERSION_TWO);
- tt_assert(pk.v2);
- tt_ptr_op(key_new_alg, OP_EQ, NULL);
- tt_ptr_op(key_new_blob, OP_EQ, NULL);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- /* Test generating a invalid key type. */
- crypto_pk_free(pk.v2); pk.v2 = NULL;
- ret = add_onion_helper_keyarg("NEW:RSA512", 0, &key_new_alg, &key_new_blob,
- &pk, &hs_version, &err_msg);
- tt_int_op(ret, OP_EQ, -1);
- tt_int_op(hs_version, OP_EQ, HS_VERSION_TWO);
- tt_assert(!pk.v2);
- tt_ptr_op(key_new_alg, OP_EQ, NULL);
- tt_ptr_op(key_new_blob, OP_EQ, NULL);
- tt_assert(err_msg);
- /* Test loading a RSA1024 key. */
- tor_free(err_msg);
- pk1 = pk_generate(0);
- tt_int_op(0, OP_EQ, crypto_pk_base64_encode_private(pk1, &encoded));
- tor_asprintf(&arg_str, "RSA1024:%s", encoded);
- ret = add_onion_helper_keyarg(arg_str, 0, &key_new_alg, &key_new_blob,
- &pk, &hs_version, &err_msg);
- tt_int_op(ret, OP_EQ, 0);
- tt_int_op(hs_version, OP_EQ, HS_VERSION_TWO);
- tt_assert(pk.v2);
- tt_ptr_op(key_new_alg, OP_EQ, NULL);
- tt_ptr_op(key_new_blob, OP_EQ, NULL);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- tt_int_op(crypto_pk_cmp_keys(pk1, pk.v2), OP_EQ, 0);
- /* Test loading a invalid key type. */
- tor_free(arg_str);
- crypto_pk_free(pk1); pk1 = NULL;
- crypto_pk_free(pk.v2); pk.v2 = NULL;
- tor_asprintf(&arg_str, "RSA512:%s", encoded);
- ret = add_onion_helper_keyarg(arg_str, 0, &key_new_alg, &key_new_blob,
- &pk, &hs_version, &err_msg);
- tt_int_op(ret, OP_EQ, -1);
- tt_int_op(hs_version, OP_EQ, HS_VERSION_TWO);
- tt_assert(!pk.v2);
- tt_ptr_op(key_new_alg, OP_EQ, NULL);
- tt_ptr_op(key_new_blob, OP_EQ, NULL);
- tt_assert(err_msg);
- /* Test loading a invalid key. */
- tor_free(arg_str);
- crypto_pk_free(pk.v2); pk.v2 = NULL;
- tor_free(err_msg);
- encoded[strlen(encoded)/2] = '\0';
- tor_asprintf(&arg_str, "RSA1024:%s", encoded);
- ret = add_onion_helper_keyarg(arg_str, 0, &key_new_alg, &key_new_blob,
- &pk, &hs_version, &err_msg);
- tt_int_op(ret, OP_EQ, -1);
- tt_int_op(hs_version, OP_EQ, HS_VERSION_TWO);
- tt_assert(!pk.v2);
- tt_ptr_op(key_new_alg, OP_EQ, NULL);
- tt_ptr_op(key_new_blob, OP_EQ, NULL);
- tt_assert(err_msg);
- done:
- crypto_pk_free(pk1);
- crypto_pk_free(pk.v2);
- tor_free(key_new_blob);
- tor_free(err_msg);
- tor_free(encoded);
- tor_free(arg_str);
- }
- static void
- test_getinfo_helper_onion(void *arg)
- {
- (void)arg;
- control_connection_t dummy;
- /* Get results out */
- char *answer = NULL;
- const char *errmsg = NULL;
- char *service_id = NULL;
- int rt = 0;
- dummy.ephemeral_onion_services = NULL;
- /* successfully get an empty answer */
- rt = getinfo_helper_onions(&dummy, "onions/current", &answer, &errmsg);
- tt_int_op(rt, OP_EQ, 0);
- tt_str_op(answer, OP_EQ, "");
- tor_free(answer);
- /* successfully get an empty answer */
- rt = getinfo_helper_onions(&dummy, "onions/detached", &answer, &errmsg);
- tt_int_op(rt, OP_EQ, 0);
- tt_str_op(answer, OP_EQ, "");
- tor_free(answer);
- /* get an answer for one onion service */
- service_id = tor_strdup("dummy_onion_id");
- dummy.ephemeral_onion_services = smartlist_new();
- smartlist_add(dummy.ephemeral_onion_services, service_id);
- rt = getinfo_helper_onions(&dummy, "onions/current", &answer, &errmsg);
- tt_int_op(rt, OP_EQ, 0);
- tt_str_op(answer, OP_EQ, "dummy_onion_id");
- done:
- tor_free(answer);
- tor_free(service_id);
- smartlist_free(dummy.ephemeral_onion_services);
- }
- static void
- test_rend_service_parse_port_config(void *arg)
- {
- const char *sep = ",";
- rend_service_port_config_t *cfg = NULL;
- char *err_msg = NULL;
- (void)arg;
- /* Test "VIRTPORT" only. */
- cfg = rend_service_parse_port_config("80", sep, &err_msg);
- tt_assert(cfg);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- /* Test "VIRTPORT,TARGET" (Target is port). */
- rend_service_port_config_free(cfg);
- cfg = rend_service_parse_port_config("80,8080", sep, &err_msg);
- tt_assert(cfg);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- /* Test "VIRTPORT,TARGET" (Target is IPv4:port). */
- rend_service_port_config_free(cfg);
- cfg = rend_service_parse_port_config("80,192.0.2.1:8080", sep, &err_msg);
- tt_assert(cfg);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- /* Test "VIRTPORT,TARGET" (Target is IPv6:port). */
- rend_service_port_config_free(cfg);
- cfg = rend_service_parse_port_config("80,[2001:db8::1]:8080", sep, &err_msg);
- tt_assert(cfg);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- rend_service_port_config_free(cfg);
- cfg = NULL;
- /* XXX: Someone should add tests for AF_UNIX targets if supported. */
- /* Test empty config. */
- rend_service_port_config_free(cfg);
- cfg = rend_service_parse_port_config("", sep, &err_msg);
- tt_ptr_op(cfg, OP_EQ, NULL);
- tt_assert(err_msg);
- /* Test invalid port. */
- tor_free(err_msg);
- cfg = rend_service_parse_port_config("90001", sep, &err_msg);
- tt_ptr_op(cfg, OP_EQ, NULL);
- tt_assert(err_msg);
- tor_free(err_msg);
- /* unix port */
- cfg = NULL;
- /* quoted unix port */
- tor_free(err_msg);
- cfg = rend_service_parse_port_config("100 unix:\"/tmp/foo bar\"",
- " ", &err_msg);
- tt_assert(cfg);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- rend_service_port_config_free(cfg);
- cfg = NULL;
- /* quoted unix port */
- tor_free(err_msg);
- cfg = rend_service_parse_port_config("100 unix:\"/tmp/foo bar\"",
- " ", &err_msg);
- tt_assert(cfg);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- rend_service_port_config_free(cfg);
- cfg = NULL;
- /* quoted unix port, missing end quote */
- cfg = rend_service_parse_port_config("100 unix:\"/tmp/foo bar",
- " ", &err_msg);
- tt_ptr_op(cfg, OP_EQ, NULL);
- tt_str_op(err_msg, OP_EQ, "Couldn't process address <unix:\"/tmp/foo bar> "
- "from hidden service configuration");
- tor_free(err_msg);
- /* bogus IP address */
- MOCK(tor_addr_lookup, mock_tor_addr_lookup__fail_on_bad_addrs);
- cfg = rend_service_parse_port_config("100 foo!!.example.com:9000",
- " ", &err_msg);
- UNMOCK(tor_addr_lookup);
- tt_ptr_op(cfg, OP_EQ, NULL);
- tt_str_op(err_msg, OP_EQ, "Unparseable address in hidden service port "
- "configuration.");
- tor_free(err_msg);
- /* bogus port port */
- cfg = rend_service_parse_port_config("100 99999",
- " ", &err_msg);
- tt_ptr_op(cfg, OP_EQ, NULL);
- tt_str_op(err_msg, OP_EQ, "Unparseable or out-of-range port \"99999\" "
- "in hidden service port configuration.");
- tor_free(err_msg);
- /* Wrong target address and port separation */
- cfg = rend_service_parse_port_config("80,127.0.0.1 1234", sep,
- &err_msg);
- tt_ptr_op(cfg, OP_EQ, NULL);
- tt_assert(err_msg);
- tor_free(err_msg);
- done:
- rend_service_port_config_free(cfg);
- tor_free(err_msg);
- }
- static void
- test_add_onion_helper_clientauth(void *arg)
- {
- rend_authorized_client_t *client = NULL;
- char *err_msg = NULL;
- int created = 0;
- (void)arg;
- /* Test "ClientName" only. */
- client = add_onion_helper_clientauth("alice", &created, &err_msg);
- tt_assert(client);
- tt_assert(created);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- rend_authorized_client_free(client);
- /* Test "ClientName:Blob" */
- client = add_onion_helper_clientauth("alice:475hGBHPlq7Mc0cRZitK/B",
- &created, &err_msg);
- tt_assert(client);
- tt_assert(!created);
- tt_ptr_op(err_msg, OP_EQ, NULL);
- rend_authorized_client_free(client);
- /* Test invalid client names */
- client = add_onion_helper_clientauth("no*asterisks*allowed", &created,
- &err_msg);
- tt_ptr_op(client, OP_EQ, NULL);
- tt_assert(err_msg);
- tor_free(err_msg);
- /* Test invalid auth cookie */
- client = add_onion_helper_clientauth("alice:12345", &created, &err_msg);
- tt_ptr_op(client, OP_EQ, NULL);
- tt_assert(err_msg);
- tor_free(err_msg);
- /* Test invalid syntax */
- client = add_onion_helper_clientauth(":475hGBHPlq7Mc0cRZitK/B", &created,
- &err_msg);
- tt_ptr_op(client, OP_EQ, NULL);
- tt_assert(err_msg);
- tor_free(err_msg);
- done:
- rend_authorized_client_free(client);
- tor_free(err_msg);
- }
- /* Mocks and data/variables used for GETINFO download status tests */
- static const download_status_t dl_status_default =
- { 0, 0, 0, DL_SCHED_CONSENSUS, DL_WANT_ANY_DIRSERVER,
- DL_SCHED_INCREMENT_FAILURE, 0, 0 };
- static download_status_t ns_dl_status[N_CONSENSUS_FLAVORS];
- static download_status_t ns_dl_status_bootstrap[N_CONSENSUS_FLAVORS];
- static download_status_t ns_dl_status_running[N_CONSENSUS_FLAVORS];
- /*
- * These should explore all the possible cases of download_status_to_string()
- * in control.c
- */
- static const download_status_t dls_sample_1 =
- { 1467163900, 0, 0, DL_SCHED_GENERIC, DL_WANT_ANY_DIRSERVER,
- DL_SCHED_INCREMENT_FAILURE, 0, 0 };
- static const char * dls_sample_1_str =
- "next-attempt-at 2016-06-29 01:31:40\n"
- "n-download-failures 0\n"
- "n-download-attempts 0\n"
- "schedule DL_SCHED_GENERIC\n"
- "want-authority DL_WANT_ANY_DIRSERVER\n"
- "increment-on DL_SCHED_INCREMENT_FAILURE\n"
- "backoff DL_SCHED_RANDOM_EXPONENTIAL\n"
- "last-backoff-position 0\n"
- "last-delay-used 0\n";
- static const download_status_t dls_sample_2 =
- { 1467164400, 1, 2, DL_SCHED_CONSENSUS, DL_WANT_AUTHORITY,
- DL_SCHED_INCREMENT_FAILURE, 0, 0 };
- static const char * dls_sample_2_str =
- "next-attempt-at 2016-06-29 01:40:00\n"
- "n-download-failures 1\n"
- "n-download-attempts 2\n"
- "schedule DL_SCHED_CONSENSUS\n"
- "want-authority DL_WANT_AUTHORITY\n"
- "increment-on DL_SCHED_INCREMENT_FAILURE\n"
- "backoff DL_SCHED_RANDOM_EXPONENTIAL\n"
- "last-backoff-position 0\n"
- "last-delay-used 0\n";
- static const download_status_t dls_sample_3 =
- { 1467154400, 12, 25, DL_SCHED_BRIDGE, DL_WANT_ANY_DIRSERVER,
- DL_SCHED_INCREMENT_ATTEMPT, 0, 0 };
- static const char * dls_sample_3_str =
- "next-attempt-at 2016-06-28 22:53:20\n"
- "n-download-failures 12\n"
- "n-download-attempts 25\n"
- "schedule DL_SCHED_BRIDGE\n"
- "want-authority DL_WANT_ANY_DIRSERVER\n"
- "increment-on DL_SCHED_INCREMENT_ATTEMPT\n"
- "backoff DL_SCHED_RANDOM_EXPONENTIAL\n"
- "last-backoff-position 0\n"
- "last-delay-used 0\n";
- static const download_status_t dls_sample_4 =
- { 1467166600, 3, 0, DL_SCHED_GENERIC, DL_WANT_ANY_DIRSERVER,
- DL_SCHED_INCREMENT_FAILURE, 0, 0 };
- static const char * dls_sample_4_str =
- "next-attempt-at 2016-06-29 02:16:40\n"
- "n-download-failures 3\n"
- "n-download-attempts 0\n"
- "schedule DL_SCHED_GENERIC\n"
- "want-authority DL_WANT_ANY_DIRSERVER\n"
- "increment-on DL_SCHED_INCREMENT_FAILURE\n"
- "backoff DL_SCHED_RANDOM_EXPONENTIAL\n"
- "last-backoff-position 0\n"
- "last-delay-used 0\n";
- static const download_status_t dls_sample_5 =
- { 1467164600, 3, 7, DL_SCHED_CONSENSUS, DL_WANT_ANY_DIRSERVER,
- DL_SCHED_INCREMENT_FAILURE, 1, 2112, };
- static const char * dls_sample_5_str =
- "next-attempt-at 2016-06-29 01:43:20\n"
- "n-download-failures 3\n"
- "n-download-attempts 7\n"
- "schedule DL_SCHED_CONSENSUS\n"
- "want-authority DL_WANT_ANY_DIRSERVER\n"
- "increment-on DL_SCHED_INCREMENT_FAILURE\n"
- "backoff DL_SCHED_RANDOM_EXPONENTIAL\n"
- "last-backoff-position 1\n"
- "last-delay-used 2112\n";
- static const download_status_t dls_sample_6 =
- { 1467164200, 4, 9, DL_SCHED_CONSENSUS, DL_WANT_AUTHORITY,
- DL_SCHED_INCREMENT_ATTEMPT, 3, 432 };
- static const char * dls_sample_6_str =
- "next-attempt-at 2016-06-29 01:36:40\n"
- "n-download-failures 4\n"
- "n-download-attempts 9\n"
- "schedule DL_SCHED_CONSENSUS\n"
- "want-authority DL_WANT_AUTHORITY\n"
- "increment-on DL_SCHED_INCREMENT_ATTEMPT\n"
- "backoff DL_SCHED_RANDOM_EXPONENTIAL\n"
- "last-backoff-position 3\n"
- "last-delay-used 432\n";
- /* Simulated auth certs */
- static const char *auth_id_digest_1_str =
- "63CDD326DFEF0CA020BDD3FEB45A3286FE13A061";
- static download_status_t auth_def_cert_download_status_1;
- static const char *auth_id_digest_2_str =
- "2C209FCDD8D48DC049777B8DC2C0F94A0408BE99";
- static download_status_t auth_def_cert_download_status_2;
- /* Expected form of digest list returned for GETINFO downloads/cert/fps */
- static const char *auth_id_digest_expected_list =
- "63CDD326DFEF0CA020BDD3FEB45A3286FE13A061\n"
- "2C209FCDD8D48DC049777B8DC2C0F94A0408BE99\n";
- /* Signing keys for simulated auth 1 */
- static const char *auth_1_sk_1_str =
- "AA69566029B1F023BA09451B8F1B10952384EB58";
- static download_status_t auth_1_sk_1_dls;
- static const char *auth_1_sk_2_str =
- "710865C7F06B73C5292695A8C34F1C94F769FF72";
- static download_status_t auth_1_sk_2_dls;
- /*
- * Expected form of sk digest list for
- * GETINFO downloads/cert/<auth_id_digest_1_str>/sks
- */
- static const char *auth_1_sk_digest_expected_list =
- "AA69566029B1F023BA09451B8F1B10952384EB58\n"
- "710865C7F06B73C5292695A8C34F1C94F769FF72\n";
- /* Signing keys for simulated auth 2 */
- static const char *auth_2_sk_1_str =
- "4299047E00D070AD6703FE00BE7AA756DB061E62";
- static download_status_t auth_2_sk_1_dls;
- static const char *auth_2_sk_2_str =
- "9451B8F1B10952384EB58B5F230C0BB701626C9B";
- static download_status_t auth_2_sk_2_dls;
- /*
- * Expected form of sk digest list for
- * GETINFO downloads/cert/<auth_id_digest_2_str>/sks
- */
- static const char *auth_2_sk_digest_expected_list =
- "4299047E00D070AD6703FE00BE7AA756DB061E62\n"
- "9451B8F1B10952384EB58B5F230C0BB701626C9B\n";
- /* Simulated router descriptor digests or bridge identity digests */
- static const char *descbr_digest_1_str =
- "616408544C7345822696074A1A3DFA16AB381CBD";
- static download_status_t descbr_digest_1_dl;
- static const char *descbr_digest_2_str =
- "06E8067246967265DBCB6641631B530EFEC12DC3";
- static download_status_t descbr_digest_2_dl;
- /* Expected form of digest list returned for GETINFO downloads/desc/descs */
- static const char *descbr_expected_list =
- "616408544C7345822696074A1A3DFA16AB381CBD\n"
- "06E8067246967265DBCB6641631B530EFEC12DC3\n";
- /*
- * Flag to make all descbr queries fail, to simulate not being
- * configured such that such queries make sense.
- */
- static int disable_descbr = 0;
- static void
- reset_mocked_dl_statuses(void)
- {
- int i;
- for (i = 0; i < N_CONSENSUS_FLAVORS; ++i) {
- memcpy(&(ns_dl_status[i]), &dl_status_default,
- sizeof(download_status_t));
- memcpy(&(ns_dl_status_bootstrap[i]), &dl_status_default,
- sizeof(download_status_t));
- memcpy(&(ns_dl_status_running[i]), &dl_status_default,
- sizeof(download_status_t));
- }
- memcpy(&auth_def_cert_download_status_1, &dl_status_default,
- sizeof(download_status_t));
- memcpy(&auth_def_cert_download_status_2, &dl_status_default,
- sizeof(download_status_t));
- memcpy(&auth_1_sk_1_dls, &dl_status_default,
- sizeof(download_status_t));
- memcpy(&auth_1_sk_2_dls, &dl_status_default,
- sizeof(download_status_t));
- memcpy(&auth_2_sk_1_dls, &dl_status_default,
- sizeof(download_status_t));
- memcpy(&auth_2_sk_2_dls, &dl_status_default,
- sizeof(download_status_t));
- memcpy(&descbr_digest_1_dl, &dl_status_default,
- sizeof(download_status_t));
- memcpy(&descbr_digest_2_dl, &dl_status_default,
- sizeof(download_status_t));
- }
- static download_status_t *
- ns_dl_status_mock(consensus_flavor_t flavor)
- {
- return &(ns_dl_status[flavor]);
- }
- static download_status_t *
- ns_dl_status_bootstrap_mock(consensus_flavor_t flavor)
- {
- return &(ns_dl_status_bootstrap[flavor]);
- }
- static download_status_t *
- ns_dl_status_running_mock(consensus_flavor_t flavor)
- {
- return &(ns_dl_status_running[flavor]);
- }
- static void
- setup_ns_mocks(void)
- {
- MOCK(networkstatus_get_dl_status_by_flavor, ns_dl_status_mock);
- MOCK(networkstatus_get_dl_status_by_flavor_bootstrap,
- ns_dl_status_bootstrap_mock);
- MOCK(networkstatus_get_dl_status_by_flavor_running,
- ns_dl_status_running_mock);
- reset_mocked_dl_statuses();
- }
- static void
- clear_ns_mocks(void)
- {
- UNMOCK(networkstatus_get_dl_status_by_flavor);
- UNMOCK(networkstatus_get_dl_status_by_flavor_bootstrap);
- UNMOCK(networkstatus_get_dl_status_by_flavor_running);
- }
- static smartlist_t *
- cert_dl_status_auth_ids_mock(void)
- {
- char digest[DIGEST_LEN], *tmp;
- int len;
- smartlist_t *list = NULL;
- /* Just pretend we have only the two hard-coded digests listed above */
- list = smartlist_new();
- len = base16_decode(digest, DIGEST_LEN,
- auth_id_digest_1_str, strlen(auth_id_digest_1_str));
- tt_int_op(len, OP_EQ, DIGEST_LEN);
- tmp = tor_malloc(DIGEST_LEN);
- memcpy(tmp, digest, DIGEST_LEN);
- smartlist_add(list, tmp);
- len = base16_decode(digest, DIGEST_LEN,
- auth_id_digest_2_str, strlen(auth_id_digest_2_str));
- tt_int_op(len, OP_EQ, DIGEST_LEN);
- tmp = tor_malloc(DIGEST_LEN);
- memcpy(tmp, digest, DIGEST_LEN);
- smartlist_add(list, tmp);
- done:
- return list;
- }
- static download_status_t *
- cert_dl_status_def_for_auth_mock(const char *digest)
- {
- download_status_t *dl = NULL;
- char digest_str[HEX_DIGEST_LEN+1];
- tt_ptr_op(digest, OP_NE, NULL);
- base16_encode(digest_str, HEX_DIGEST_LEN + 1,
- digest, DIGEST_LEN);
- digest_str[HEX_DIGEST_LEN] = '\0';
- if (strcmp(digest_str, auth_id_digest_1_str) == 0) {
- dl = &auth_def_cert_download_status_1;
- } else if (strcmp(digest_str, auth_id_digest_2_str) == 0) {
- dl = &auth_def_cert_download_status_2;
- }
- done:
- return dl;
- }
- static smartlist_t *
- cert_dl_status_sks_for_auth_id_mock(const char *digest)
- {
- smartlist_t *list = NULL;
- char sk[DIGEST_LEN];
- char digest_str[HEX_DIGEST_LEN+1];
- char *tmp;
- int len;
- tt_ptr_op(digest, OP_NE, NULL);
- base16_encode(digest_str, HEX_DIGEST_LEN + 1,
- digest, DIGEST_LEN);
- digest_str[HEX_DIGEST_LEN] = '\0';
- /*
- * Build a list of two hard-coded digests, depending on what we
- * were just passed.
- */
- if (strcmp(digest_str, auth_id_digest_1_str) == 0) {
- list = smartlist_new();
- len = base16_decode(sk, DIGEST_LEN,
- auth_1_sk_1_str, strlen(auth_1_sk_1_str));
- tt_int_op(len, OP_EQ, DIGEST_LEN);
- tmp = tor_malloc(DIGEST_LEN);
- memcpy(tmp, sk, DIGEST_LEN);
- smartlist_add(list, tmp);
- len = base16_decode(sk, DIGEST_LEN,
- auth_1_sk_2_str, strlen(auth_1_sk_2_str));
- tt_int_op(len, OP_EQ, DIGEST_LEN);
- tmp = tor_malloc(DIGEST_LEN);
- memcpy(tmp, sk, DIGEST_LEN);
- smartlist_add(list, tmp);
- } else if (strcmp(digest_str, auth_id_digest_2_str) == 0) {
- list = smartlist_new();
- len = base16_decode(sk, DIGEST_LEN,
- auth_2_sk_1_str, strlen(auth_2_sk_1_str));
- tt_int_op(len, OP_EQ, DIGEST_LEN);
- tmp = tor_malloc(DIGEST_LEN);
- memcpy(tmp, sk, DIGEST_LEN);
- smartlist_add(list, tmp);
- len = base16_decode(sk, DIGEST_LEN,
- auth_2_sk_2_str, strlen(auth_2_sk_2_str));
- tt_int_op(len, OP_EQ, DIGEST_LEN);
- tmp = tor_malloc(DIGEST_LEN);
- memcpy(tmp, sk, DIGEST_LEN);
- smartlist_add(list, tmp);
- }
- done:
- return list;
- }
- static download_status_t *
- cert_dl_status_fp_sk_mock(const char *fp_digest, const char *sk_digest)
- {
- download_status_t *dl = NULL;
- char fp_digest_str[HEX_DIGEST_LEN+1], sk_digest_str[HEX_DIGEST_LEN+1];
- /*
- * Unpack the digests so we can compare them and figure out which
- * dl status we want.
- */
- tt_ptr_op(fp_digest, OP_NE, NULL);
- base16_encode(fp_digest_str, HEX_DIGEST_LEN + 1,
- fp_digest, DIGEST_LEN);
- fp_digest_str[HEX_DIGEST_LEN] = '\0';
- tt_ptr_op(sk_digest, OP_NE, NULL);
- base16_encode(sk_digest_str, HEX_DIGEST_LEN + 1,
- sk_digest, DIGEST_LEN);
- sk_digest_str[HEX_DIGEST_LEN] = '\0';
- if (strcmp(fp_digest_str, auth_id_digest_1_str) == 0) {
- if (strcmp(sk_digest_str, auth_1_sk_1_str) == 0) {
- dl = &auth_1_sk_1_dls;
- } else if (strcmp(sk_digest_str, auth_1_sk_2_str) == 0) {
- dl = &auth_1_sk_2_dls;
- }
- } else if (strcmp(fp_digest_str, auth_id_digest_2_str) == 0) {
- if (strcmp(sk_digest_str, auth_2_sk_1_str) == 0) {
- dl = &auth_2_sk_1_dls;
- } else if (strcmp(sk_digest_str, auth_2_sk_2_str) == 0) {
- dl = &auth_2_sk_2_dls;
- }
- }
- done:
- return dl;
- }
- static void
- setup_cert_mocks(void)
- {
- MOCK(list_authority_ids_with_downloads, cert_dl_status_auth_ids_mock);
- MOCK(id_only_download_status_for_authority_id,
- cert_dl_status_def_for_auth_mock);
- MOCK(list_sk_digests_for_authority_id,
- cert_dl_status_sks_for_auth_id_mock);
- MOCK(download_status_for_authority_id_and_sk,
- cert_dl_status_fp_sk_mock);
- reset_mocked_dl_statuses();
- }
- static void
- clear_cert_mocks(void)
- {
- UNMOCK(list_authority_ids_with_downloads);
- UNMOCK(id_only_download_status_for_authority_id);
- UNMOCK(list_sk_digests_for_authority_id);
- UNMOCK(download_status_for_authority_id_and_sk);
- }
- static smartlist_t *
- descbr_get_digests_mock(void)
- {
- char digest[DIGEST_LEN], *tmp;
- int len;
- smartlist_t *list = NULL;
- if (!disable_descbr) {
- /* Just pretend we have only the two hard-coded digests listed above */
- list = smartlist_new();
- len = base16_decode(digest, DIGEST_LEN,
- descbr_digest_1_str, strlen(descbr_digest_1_str));
- tt_int_op(len, OP_EQ, DIGEST_LEN);
- tmp = tor_malloc(DIGEST_LEN);
- memcpy(tmp, digest, DIGEST_LEN);
- smartlist_add(list, tmp);
- len = base16_decode(digest, DIGEST_LEN,
- descbr_digest_2_str, strlen(descbr_digest_2_str));
- tt_int_op(len, OP_EQ, DIGEST_LEN);
- tmp = tor_malloc(DIGEST_LEN);
- memcpy(tmp, digest, DIGEST_LEN);
- smartlist_add(list, tmp);
- }
- done:
- return list;
- }
- static download_status_t *
- descbr_get_dl_by_digest_mock(const char *digest)
- {
- download_status_t *dl = NULL;
- char digest_str[HEX_DIGEST_LEN+1];
- if (!disable_descbr) {
- tt_ptr_op(digest, OP_NE, NULL);
- base16_encode(digest_str, HEX_DIGEST_LEN + 1,
- digest, DIGEST_LEN);
- digest_str[HEX_DIGEST_LEN] = '\0';
- if (strcmp(digest_str, descbr_digest_1_str) == 0) {
- dl = &descbr_digest_1_dl;
- } else if (strcmp(digest_str, descbr_digest_2_str) == 0) {
- dl = &descbr_digest_2_dl;
- }
- }
- done:
- return dl;
- }
- static void
- setup_desc_mocks(void)
- {
- MOCK(router_get_descriptor_digests,
- descbr_get_digests_mock);
- MOCK(router_get_dl_status_by_descriptor_digest,
- descbr_get_dl_by_digest_mock);
- reset_mocked_dl_statuses();
- }
- static void
- clear_desc_mocks(void)
- {
- UNMOCK(router_get_descriptor_digests);
- UNMOCK(router_get_dl_status_by_descriptor_digest);
- }
- static void
- setup_bridge_mocks(void)
- {
- disable_descbr = 0;
- MOCK(list_bridge_identities,
- descbr_get_digests_mock);
- MOCK(get_bridge_dl_status_by_id,
- descbr_get_dl_by_digest_mock);
- reset_mocked_dl_statuses();
- }
- static void
- clear_bridge_mocks(void)
- {
- UNMOCK(list_bridge_identities);
- UNMOCK(get_bridge_dl_status_by_id);
- disable_descbr = 0;
- }
- static void
- test_download_status_consensus(void *arg)
- {
- /* We just need one of these to pass, it doesn't matter what's in it */
- control_connection_t dummy;
- /* Get results out */
- char *answer = NULL;
- const char *errmsg = NULL;
- (void)arg;
- /* Check that the unknown prefix case works; no mocks needed yet */
- getinfo_helper_downloads(&dummy, "downloads/foo", &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_str_op(errmsg, OP_EQ, "Unknown download status query");
- setup_ns_mocks();
- /*
- * Check returning serialized dlstatuses, and implicitly also test
- * download_status_to_string().
- */
- /* Case 1 default/FLAV_NS*/
- memcpy(&(ns_dl_status[FLAV_NS]), &dls_sample_1,
- sizeof(download_status_t));
- getinfo_helper_downloads(&dummy, "downloads/networkstatus/ns",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_1_str);
- tor_free(answer);
- errmsg = NULL;
- /* Case 2 default/FLAV_MICRODESC */
- memcpy(&(ns_dl_status[FLAV_MICRODESC]), &dls_sample_2,
- sizeof(download_status_t));
- getinfo_helper_downloads(&dummy, "downloads/networkstatus/microdesc",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_2_str);
- tor_free(answer);
- errmsg = NULL;
- /* Case 3 bootstrap/FLAV_NS */
- memcpy(&(ns_dl_status_bootstrap[FLAV_NS]), &dls_sample_3,
- sizeof(download_status_t));
- getinfo_helper_downloads(&dummy, "downloads/networkstatus/ns/bootstrap",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_3_str);
- tor_free(answer);
- errmsg = NULL;
- /* Case 4 bootstrap/FLAV_MICRODESC */
- memcpy(&(ns_dl_status_bootstrap[FLAV_MICRODESC]), &dls_sample_4,
- sizeof(download_status_t));
- getinfo_helper_downloads(&dummy,
- "downloads/networkstatus/microdesc/bootstrap",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_4_str);
- tor_free(answer);
- errmsg = NULL;
- /* Case 5 running/FLAV_NS */
- memcpy(&(ns_dl_status_running[FLAV_NS]), &dls_sample_5,
- sizeof(download_status_t));
- getinfo_helper_downloads(&dummy,
- "downloads/networkstatus/ns/running",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_5_str);
- tor_free(answer);
- errmsg = NULL;
- /* Case 6 running/FLAV_MICRODESC */
- memcpy(&(ns_dl_status_running[FLAV_MICRODESC]), &dls_sample_6,
- sizeof(download_status_t));
- getinfo_helper_downloads(&dummy,
- "downloads/networkstatus/microdesc/running",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_6_str);
- tor_free(answer);
- errmsg = NULL;
- /* Now check the error case */
- getinfo_helper_downloads(&dummy, "downloads/networkstatus/foo",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "Unknown flavor");
- errmsg = NULL;
- done:
- clear_ns_mocks();
- tor_free(answer);
- return;
- }
- static void
- test_download_status_cert(void *arg)
- {
- /* We just need one of these to pass, it doesn't matter what's in it */
- control_connection_t dummy;
- /* Get results out */
- char *question = NULL;
- char *answer = NULL;
- const char *errmsg = NULL;
- (void)arg;
- setup_cert_mocks();
- /*
- * Check returning serialized dlstatuses and digest lists, and implicitly
- * also test download_status_to_string() and digest_list_to_string().
- */
- /* Case 1 - list of authority identity fingerprints */
- getinfo_helper_downloads(&dummy,
- "downloads/cert/fps",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, auth_id_digest_expected_list);
- tor_free(answer);
- errmsg = NULL;
- /* Case 2 - download status for default cert for 1st auth id */
- memcpy(&auth_def_cert_download_status_1, &dls_sample_1,
- sizeof(download_status_t));
- tor_asprintf(&question, "downloads/cert/fp/%s", auth_id_digest_1_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_1_str);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Case 3 - download status for default cert for 2nd auth id */
- memcpy(&auth_def_cert_download_status_2, &dls_sample_2,
- sizeof(download_status_t));
- tor_asprintf(&question, "downloads/cert/fp/%s", auth_id_digest_2_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_2_str);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Case 4 - list of signing key digests for 1st auth id */
- tor_asprintf(&question, "downloads/cert/fp/%s/sks", auth_id_digest_1_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, auth_1_sk_digest_expected_list);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Case 5 - list of signing key digests for 2nd auth id */
- tor_asprintf(&question, "downloads/cert/fp/%s/sks", auth_id_digest_2_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, auth_2_sk_digest_expected_list);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Case 6 - download status for 1st auth id, 1st sk */
- memcpy(&auth_1_sk_1_dls, &dls_sample_3,
- sizeof(download_status_t));
- tor_asprintf(&question, "downloads/cert/fp/%s/%s",
- auth_id_digest_1_str, auth_1_sk_1_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_3_str);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Case 7 - download status for 1st auth id, 2nd sk */
- memcpy(&auth_1_sk_2_dls, &dls_sample_4,
- sizeof(download_status_t));
- tor_asprintf(&question, "downloads/cert/fp/%s/%s",
- auth_id_digest_1_str, auth_1_sk_2_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_4_str);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Case 8 - download status for 2nd auth id, 1st sk */
- memcpy(&auth_2_sk_1_dls, &dls_sample_5,
- sizeof(download_status_t));
- tor_asprintf(&question, "downloads/cert/fp/%s/%s",
- auth_id_digest_2_str, auth_2_sk_1_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_5_str);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Case 9 - download status for 2nd auth id, 2nd sk */
- memcpy(&auth_2_sk_2_dls, &dls_sample_6,
- sizeof(download_status_t));
- tor_asprintf(&question, "downloads/cert/fp/%s/%s",
- auth_id_digest_2_str, auth_2_sk_2_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_6_str);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Now check the error cases */
- /* Case 1 - query is garbage after downloads/cert/ part */
- getinfo_helper_downloads(&dummy, "downloads/cert/blahdeblah",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "Unknown certificate download status query");
- errmsg = NULL;
- /*
- * Case 2 - looks like downloads/cert/fp/<fp>, but <fp> isn't even
- * the right length for a digest.
- */
- getinfo_helper_downloads(&dummy, "downloads/cert/fp/2B1D36D32B2942406",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "That didn't look like a digest");
- errmsg = NULL;
- /*
- * Case 3 - looks like downloads/cert/fp/<fp>, and <fp> is digest-sized,
- * but not parseable as one.
- */
- getinfo_helper_downloads(&dummy,
- "downloads/cert/fp/82F52AF55D250115FE44D3GC81D49643241D56A1",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "That didn't look like a digest");
- errmsg = NULL;
- /*
- * Case 4 - downloads/cert/fp/<fp>, and <fp> is not a known authority
- * identity digest
- */
- getinfo_helper_downloads(&dummy,
- "downloads/cert/fp/AC4F23B5745BDD2A77997B85B1FD85D05C2E0F61",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ,
- "Failed to get download status for this authority identity digest");
- errmsg = NULL;
- /*
- * Case 5 - looks like downloads/cert/fp/<fp>/<anything>, but <fp> doesn't
- * parse as a sensible digest.
- */
- getinfo_helper_downloads(&dummy,
- "downloads/cert/fp/82F52AF55D250115FE44D3GC81D49643241D56A1/blah",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "That didn't look like an identity digest");
- errmsg = NULL;
- /*
- * Case 6 - looks like downloads/cert/fp/<fp>/<anything>, but <fp> doesn't
- * parse as a sensible digest.
- */
- getinfo_helper_downloads(&dummy,
- "downloads/cert/fp/82F52AF55D25/blah",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "That didn't look like an identity digest");
- errmsg = NULL;
- /*
- * Case 7 - downloads/cert/fp/<fp>/sks, and <fp> is not a known authority
- * digest.
- */
- getinfo_helper_downloads(&dummy,
- "downloads/cert/fp/AC4F23B5745BDD2A77997B85B1FD85D05C2E0F61/sks",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ,
- "Failed to get list of signing key digests for this authority "
- "identity digest");
- errmsg = NULL;
- /*
- * Case 8 - looks like downloads/cert/fp/<fp>/<sk>, but <sk> doesn't
- * parse as a signing key digest.
- */
- getinfo_helper_downloads(&dummy,
- "downloads/cert/fp/AC4F23B5745BDD2A77997B85B1FD85D05C2E0F61/"
- "82F52AF55D250115FE44D3GC81D49643241D56A1",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "That didn't look like a signing key digest");
- errmsg = NULL;
- /*
- * Case 9 - looks like downloads/cert/fp/<fp>/<sk>, but <sk> doesn't
- * parse as a signing key digest.
- */
- getinfo_helper_downloads(&dummy,
- "downloads/cert/fp/AC4F23B5745BDD2A77997B85B1FD85D05C2E0F61/"
- "82F52AF55D250115FE44D",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "That didn't look like a signing key digest");
- errmsg = NULL;
- /*
- * Case 10 - downloads/cert/fp/<fp>/<sk>, but <fp> isn't a known
- * authority identity digest.
- */
- getinfo_helper_downloads(&dummy,
- "downloads/cert/fp/C6B05DF332F74DB9A13498EE3BBC7AA2F69FCB45/"
- "3A214FC21AE25B012C2ECCB5F4EC8A3602D0545D",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ,
- "Failed to get download status for this identity/"
- "signing key digest pair");
- errmsg = NULL;
- /*
- * Case 11 - downloads/cert/fp/<fp>/<sk>, but <sk> isn't a known
- * signing key digest.
- */
- getinfo_helper_downloads(&dummy,
- "downloads/cert/fp/63CDD326DFEF0CA020BDD3FEB45A3286FE13A061/"
- "3A214FC21AE25B012C2ECCB5F4EC8A3602D0545D",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ,
- "Failed to get download status for this identity/"
- "signing key digest pair");
- errmsg = NULL;
- /*
- * Case 12 - downloads/cert/fp/<fp>/<sk>, but <sk> is on the list for
- * a different authority identity digest.
- */
- getinfo_helper_downloads(&dummy,
- "downloads/cert/fp/63CDD326DFEF0CA020BDD3FEB45A3286FE13A061/"
- "9451B8F1B10952384EB58B5F230C0BB701626C9B",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ,
- "Failed to get download status for this identity/"
- "signing key digest pair");
- errmsg = NULL;
- done:
- clear_cert_mocks();
- tor_free(answer);
- return;
- }
- static void
- test_download_status_desc(void *arg)
- {
- /* We just need one of these to pass, it doesn't matter what's in it */
- control_connection_t dummy;
- /* Get results out */
- char *question = NULL;
- char *answer = NULL;
- const char *errmsg = NULL;
- (void)arg;
- setup_desc_mocks();
- /*
- * Check returning serialized dlstatuses and digest lists, and implicitly
- * also test download_status_to_string() and digest_list_to_string().
- */
- /* Case 1 - list of router descriptor digests */
- getinfo_helper_downloads(&dummy,
- "downloads/desc/descs",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, descbr_expected_list);
- tor_free(answer);
- errmsg = NULL;
- /* Case 2 - get download status for router descriptor 1 */
- memcpy(&descbr_digest_1_dl, &dls_sample_1,
- sizeof(download_status_t));
- tor_asprintf(&question, "downloads/desc/%s", descbr_digest_1_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_1_str);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Case 3 - get download status for router descriptor 1 */
- memcpy(&descbr_digest_2_dl, &dls_sample_2,
- sizeof(download_status_t));
- tor_asprintf(&question, "downloads/desc/%s", descbr_digest_2_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_2_str);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Now check the error cases */
- /* Case 1 - non-digest-length garbage after downloads/desc */
- getinfo_helper_downloads(&dummy, "downloads/desc/blahdeblah",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "Unknown router descriptor download status query");
- errmsg = NULL;
- /* Case 2 - nonparseable digest-shaped thing */
- getinfo_helper_downloads(
- &dummy,
- "downloads/desc/774EC52FD9A5B80A6FACZE536616E8022E3470AG",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "That didn't look like a digest");
- errmsg = NULL;
- /* Case 3 - digest we have no descriptor for */
- getinfo_helper_downloads(
- &dummy,
- "downloads/desc/B05B46135B0B2C04EBE1DD6A6AE4B12D7CD2226A",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "No such descriptor digest found");
- errmsg = NULL;
- /* Case 4 - microdescs only */
- disable_descbr = 1;
- getinfo_helper_downloads(&dummy,
- "downloads/desc/descs",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ,
- "We don't seem to have a networkstatus-flavored consensus");
- errmsg = NULL;
- disable_descbr = 0;
- done:
- clear_desc_mocks();
- tor_free(answer);
- return;
- }
- static void
- test_download_status_bridge(void *arg)
- {
- /* We just need one of these to pass, it doesn't matter what's in it */
- control_connection_t dummy;
- /* Get results out */
- char *question = NULL;
- char *answer = NULL;
- const char *errmsg = NULL;
- (void)arg;
- setup_bridge_mocks();
- /*
- * Check returning serialized dlstatuses and digest lists, and implicitly
- * also test download_status_to_string() and digest_list_to_string().
- */
- /* Case 1 - list of bridge identity digests */
- getinfo_helper_downloads(&dummy,
- "downloads/bridge/bridges",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, descbr_expected_list);
- tor_free(answer);
- errmsg = NULL;
- /* Case 2 - get download status for bridge descriptor 1 */
- memcpy(&descbr_digest_1_dl, &dls_sample_3,
- sizeof(download_status_t));
- tor_asprintf(&question, "downloads/bridge/%s", descbr_digest_1_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_3_str);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Case 3 - get download status for router descriptor 1 */
- memcpy(&descbr_digest_2_dl, &dls_sample_4,
- sizeof(download_status_t));
- tor_asprintf(&question, "downloads/bridge/%s", descbr_digest_2_str);
- tt_ptr_op(question, OP_NE, NULL);
- getinfo_helper_downloads(&dummy, question, &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, dls_sample_4_str);
- tor_free(question);
- tor_free(answer);
- errmsg = NULL;
- /* Now check the error cases */
- /* Case 1 - non-digest-length garbage after downloads/bridge */
- getinfo_helper_downloads(&dummy, "downloads/bridge/blahdeblah",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "Unknown bridge descriptor download status query");
- errmsg = NULL;
- /* Case 2 - nonparseable digest-shaped thing */
- getinfo_helper_downloads(
- &dummy,
- "downloads/bridge/774EC52FD9A5B80A6FACZE536616E8022E3470AG",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "That didn't look like a digest");
- errmsg = NULL;
- /* Case 3 - digest we have no descriptor for */
- getinfo_helper_downloads(
- &dummy,
- "downloads/bridge/B05B46135B0B2C04EBE1DD6A6AE4B12D7CD2226A",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "No such bridge identity digest found");
- errmsg = NULL;
- /* Case 4 - bridges disabled */
- disable_descbr = 1;
- getinfo_helper_downloads(&dummy,
- "downloads/bridge/bridges",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_EQ, NULL);
- tt_ptr_op(errmsg, OP_NE, NULL);
- tt_str_op(errmsg, OP_EQ, "We don't seem to be using bridges");
- errmsg = NULL;
- disable_descbr = 0;
- done:
- clear_bridge_mocks();
- tor_free(answer);
- return;
- }
- /** Set timeval to a mock date and time. This is necessary
- * to make tor_gettimeofday() mockable. */
- static void
- mock_tor_gettimeofday(struct timeval *timeval)
- {
- timeval->tv_sec = 1523405073;
- timeval->tv_usec = 271645;
- }
- static void
- test_current_time(void *arg)
- {
- /* We just need one of these to pass, it doesn't matter what's in it */
- control_connection_t dummy;
- /* Get results out */
- char *answer = NULL;
- const char *errmsg = NULL;
- (void)arg;
- /* We need these for storing the (mock) time. */
- MOCK(tor_gettimeofday, mock_tor_gettimeofday);
- struct timeval now;
- tor_gettimeofday(&now);
- char timebuf[ISO_TIME_LEN+1];
- /* Case 1 - local time */
- format_local_iso_time_nospace(timebuf, (time_t)now.tv_sec);
- getinfo_helper_current_time(&dummy,
- "current-time/local",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, timebuf);
- tor_free(answer);
- errmsg = NULL;
- /* Case 2 - UTC time */
- format_iso_time_nospace(timebuf, (time_t)now.tv_sec);
- getinfo_helper_current_time(&dummy,
- "current-time/utc",
- &answer, &errmsg);
- tt_ptr_op(answer, OP_NE, NULL);
- tt_ptr_op(errmsg, OP_EQ, NULL);
- tt_str_op(answer, OP_EQ, timebuf);
- tor_free(answer);
- errmsg = NULL;
- done:
- UNMOCK(tor_gettimeofday);
- tor_free(answer);
- return;
- }
- static size_t n_nodelist_get_list = 0;
- static smartlist_t *nodes = NULL;
- static smartlist_t *
- mock_nodelist_get_list(void)
- {
- n_nodelist_get_list++;
- tor_assert(nodes);
- return nodes;
- }
- static void
- test_getinfo_md_all(void *arg)
- {
- char *answer = NULL;
- const char *errmsg = NULL;
- int retval = 0;
- (void)arg;
- node_t *node1 = tor_malloc(sizeof(node_t));
- memset(node1, 0, sizeof(node_t));
- node1->md = tor_malloc(sizeof(microdesc_t));
- memset(node1->md, 0, sizeof(microdesc_t));
- node1->md->body = tor_strdup("md1\n");
- node1->md->bodylen = 4;
- node_t *node2 = tor_malloc(sizeof(node_t));
- memset(node2, 0, sizeof(node_t));
- node2->md = tor_malloc(sizeof(microdesc_t));
- memset(node2->md, 0, sizeof(microdesc_t));
- node2->md->body = tor_strdup("md2\n");
- node2->md->bodylen = 4;
- MOCK(nodelist_get_list, mock_nodelist_get_list);
- nodes = smartlist_new();
- retval = getinfo_helper_dir(NULL, "md/all", &answer, &errmsg);
- tt_int_op(n_nodelist_get_list, OP_EQ, 1);
- tt_int_op(retval, OP_EQ, 0);
- tt_assert(answer != NULL);
- tt_assert(errmsg == NULL);
- tt_str_op(answer, OP_EQ, "");
- tor_free(answer);
- smartlist_add(nodes, node1);
- smartlist_add(nodes, node2);
- retval = getinfo_helper_dir(NULL, "md/all", &answer, &errmsg);
- tt_int_op(n_nodelist_get_list, OP_EQ, 2);
- tt_int_op(retval, OP_EQ, 0);
- tt_assert(answer != NULL);
- tt_assert(errmsg == NULL);
- tt_str_op(answer, OP_EQ, "md1\nmd2\n");
- done:
- UNMOCK(nodelist_get_list);
- tor_free(node1->md->body);
- tor_free(node1->md);
- tor_free(node1);
- tor_free(node2->md->body);
- tor_free(node2->md);
- tor_free(node2);
- tor_free(answer);
- smartlist_free(nodes);
- return;
- }
- struct testcase_t controller_tests[] = {
- { "add_onion_helper_keyarg_v2", test_add_onion_helper_keyarg_v2, 0,
- NULL, NULL },
- { "add_onion_helper_keyarg_v3", test_add_onion_helper_keyarg_v3, 0,
- NULL, NULL },
- { "getinfo_helper_onion", test_getinfo_helper_onion, 0, NULL, NULL },
- { "rend_service_parse_port_config", test_rend_service_parse_port_config, 0,
- NULL, NULL },
- { "add_onion_helper_clientauth", test_add_onion_helper_clientauth, 0, NULL,
- NULL },
- { "download_status_consensus", test_download_status_consensus, 0, NULL,
- NULL },
- { "download_status_cert", test_download_status_cert, 0, NULL,
- NULL },
- { "download_status_desc", test_download_status_desc, 0, NULL, NULL },
- { "download_status_bridge", test_download_status_bridge, 0, NULL, NULL },
- { "current_time", test_current_time, 0, NULL, NULL },
- { "getinfo_md_all", test_getinfo_md_all, 0, NULL, NULL },
- END_OF_TESTCASES
- };
|