123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916 |
- /* Copyright (c) 2001-2004, Roger Dingledine.
- * Copyright (c) 2004-2006, Roger Dingledine, Nick Mathewson.
- * Copyright (c) 2007-2014, The Tor Project, Inc. */
- /* See LICENSE for licensing information */
- #include "orconfig.h"
- #include <math.h>
- #define DIRSERV_PRIVATE
- #define DIRVOTE_PRIVATE
- #define ROUTER_PRIVATE
- #define ROUTERLIST_PRIVATE
- #define HIBERNATE_PRIVATE
- #define NETWORKSTATUS_PRIVATE
- #include "or.h"
- #include "config.h"
- #include "directory.h"
- #include "dirserv.h"
- #include "dirvote.h"
- #include "hibernate.h"
- #include "networkstatus.h"
- #include "router.h"
- #include "routerlist.h"
- #include "routerparse.h"
- #include "test.h"
- static void
- test_dir_nicknames(void *arg)
- {
- (void)arg;
- tt_assert( is_legal_nickname("a"));
- tt_assert(!is_legal_nickname(""));
- tt_assert(!is_legal_nickname("abcdefghijklmnopqrst")); /* 20 chars */
- tt_assert(!is_legal_nickname("hyphen-")); /* bad char */
- tt_assert( is_legal_nickname("abcdefghijklmnopqrs")); /* 19 chars */
- tt_assert(!is_legal_nickname("$AAAAAAAA01234AAAAAAAAAAAAAAAAAAAAAAAAAAA"));
- /* valid */
- tt_assert( is_legal_nickname_or_hexdigest(
- "$AAAAAAAA01234AAAAAAAAAAAAAAAAAAAAAAAAAAA"));
- tt_assert( is_legal_nickname_or_hexdigest(
- "$AAAAAAAA01234AAAAAAAAAAAAAAAAAAAAAAAAAAA=fred"));
- tt_assert( is_legal_nickname_or_hexdigest(
- "$AAAAAAAA01234AAAAAAAAAAAAAAAAAAAAAAAAAAA~fred"));
- /* too short */
- tt_assert(!is_legal_nickname_or_hexdigest(
- "$AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA"));
- /* illegal char */
- tt_assert(!is_legal_nickname_or_hexdigest(
- "$AAAAAAzAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA"));
- /* hex part too long */
- tt_assert(!is_legal_nickname_or_hexdigest(
- "$AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA"));
- tt_assert(!is_legal_nickname_or_hexdigest(
- "$AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA=fred"));
- /* Bad nickname */
- tt_assert(!is_legal_nickname_or_hexdigest(
- "$AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA="));
- tt_assert(!is_legal_nickname_or_hexdigest(
- "$AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA~"));
- tt_assert(!is_legal_nickname_or_hexdigest(
- "$AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA~hyphen-"));
- tt_assert(!is_legal_nickname_or_hexdigest(
- "$AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA~"
- "abcdefghijklmnoppqrst"));
- /* Bad extra char. */
- tt_assert(!is_legal_nickname_or_hexdigest(
- "$AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA!"));
- tt_assert(is_legal_nickname_or_hexdigest("xyzzy"));
- tt_assert(is_legal_nickname_or_hexdigest("abcdefghijklmnopqrs"));
- tt_assert(!is_legal_nickname_or_hexdigest("abcdefghijklmnopqrst"));
- done:
- ;
- }
- /** Run unit tests for router descriptor generation logic. */
- static void
- test_dir_formats(void *arg)
- {
- char *buf = NULL;
- char buf2[8192];
- char platform[256];
- char fingerprint[FINGERPRINT_LEN+1];
- char *pk1_str = NULL, *pk2_str = NULL, *cp;
- size_t pk1_str_len, pk2_str_len;
- routerinfo_t *r1=NULL, *r2=NULL;
- crypto_pk_t *pk1 = NULL, *pk2 = NULL;
- routerinfo_t *rp1 = NULL, *rp2 = NULL;
- addr_policy_t *ex1, *ex2;
- routerlist_t *dir1 = NULL, *dir2 = NULL;
- or_options_t *options = get_options_mutable();
- const addr_policy_t *p;
- (void)arg;
- pk1 = pk_generate(0);
- pk2 = pk_generate(1);
- tt_assert(pk1 && pk2);
- hibernate_set_state_for_testing_(HIBERNATE_STATE_LIVE);
- get_platform_str(platform, sizeof(platform));
- r1 = tor_malloc_zero(sizeof(routerinfo_t));
- r1->addr = 0xc0a80001u; /* 192.168.0.1 */
- r1->cache_info.published_on = 0;
- r1->or_port = 9000;
- r1->dir_port = 9003;
- tor_addr_parse(&r1->ipv6_addr, "1:2:3:4::");
- r1->ipv6_orport = 9999;
- r1->onion_pkey = crypto_pk_dup_key(pk1);
- r1->identity_pkey = crypto_pk_dup_key(pk2);
- r1->bandwidthrate = 1000;
- r1->bandwidthburst = 5000;
- r1->bandwidthcapacity = 10000;
- r1->exit_policy = NULL;
- r1->nickname = tor_strdup("Magri");
- r1->platform = tor_strdup(platform);
- ex1 = tor_malloc_zero(sizeof(addr_policy_t));
- ex2 = tor_malloc_zero(sizeof(addr_policy_t));
- ex1->policy_type = ADDR_POLICY_ACCEPT;
- tor_addr_from_ipv4h(&ex1->addr, 0);
- ex1->maskbits = 0;
- ex1->prt_min = ex1->prt_max = 80;
- ex2->policy_type = ADDR_POLICY_REJECT;
- tor_addr_from_ipv4h(&ex2->addr, 18<<24);
- ex2->maskbits = 8;
- ex2->prt_min = ex2->prt_max = 24;
- r2 = tor_malloc_zero(sizeof(routerinfo_t));
- r2->addr = 0x0a030201u; /* 10.3.2.1 */
- r2->platform = tor_strdup(platform);
- r2->cache_info.published_on = 5;
- r2->or_port = 9005;
- r2->dir_port = 0;
- r2->onion_pkey = crypto_pk_dup_key(pk2);
- r2->onion_curve25519_pkey = tor_malloc_zero(sizeof(curve25519_public_key_t));
- curve25519_public_from_base64(r2->onion_curve25519_pkey,
- "skyinAnvardNostarsNomoonNowindormistsorsnow");
- r2->identity_pkey = crypto_pk_dup_key(pk1);
- r2->bandwidthrate = r2->bandwidthburst = r2->bandwidthcapacity = 3000;
- r2->exit_policy = smartlist_new();
- smartlist_add(r2->exit_policy, ex1);
- smartlist_add(r2->exit_policy, ex2);
- r2->nickname = tor_strdup("Fred");
- tt_assert(!crypto_pk_write_public_key_to_string(pk1, &pk1_str,
- &pk1_str_len));
- tt_assert(!crypto_pk_write_public_key_to_string(pk2 , &pk2_str,
- &pk2_str_len));
- /* XXXX025 router_dump_to_string should really take this from ri.*/
- options->ContactInfo = tor_strdup("Magri White "
- "<magri@elsewhere.example.com>");
- buf = router_dump_router_to_string(r1, pk2);
- tor_free(options->ContactInfo);
- tt_assert(buf);
- strlcpy(buf2, "router Magri 192.168.0.1 9000 0 9003\n"
- "or-address [1:2:3:4::]:9999\n"
- "platform Tor "VERSION" on ", sizeof(buf2));
- strlcat(buf2, get_uname(), sizeof(buf2));
- strlcat(buf2, "\n"
- "protocols Link 1 2 Circuit 1\n"
- "published 1970-01-01 00:00:00\n"
- "fingerprint ", sizeof(buf2));
- tt_assert(!crypto_pk_get_fingerprint(pk2, fingerprint, 1));
- strlcat(buf2, fingerprint, sizeof(buf2));
- strlcat(buf2, "\nuptime 0\n"
- /* XXX the "0" above is hard-coded, but even if we made it reflect
- * uptime, that still wouldn't make it right, because the two
- * descriptors might be made on different seconds... hm. */
- "bandwidth 1000 5000 10000\n"
- "onion-key\n", sizeof(buf2));
- strlcat(buf2, pk1_str, sizeof(buf2));
- strlcat(buf2, "signing-key\n", sizeof(buf2));
- strlcat(buf2, pk2_str, sizeof(buf2));
- strlcat(buf2, "hidden-service-dir\n", sizeof(buf2));
- strlcat(buf2, "contact Magri White <magri@elsewhere.example.com>\n",
- sizeof(buf2));
- strlcat(buf2, "reject *:*\nrouter-signature\n", sizeof(buf2));
- buf[strlen(buf2)] = '\0'; /* Don't compare the sig; it's never the same
- * twice */
- tt_str_op(buf,OP_EQ, buf2);
- tor_free(buf);
- buf = router_dump_router_to_string(r1, pk2);
- tt_assert(buf);
- cp = buf;
- rp1 = router_parse_entry_from_string((const char*)cp,NULL,1,0,NULL,NULL);
- tt_assert(rp1);
- tt_int_op(rp1->addr,OP_EQ, r1->addr);
- tt_int_op(rp1->or_port,OP_EQ, r1->or_port);
- //test_eq(rp1->dir_port, r1->dir_port);
- tt_int_op(rp1->bandwidthrate,OP_EQ, r1->bandwidthrate);
- tt_int_op(rp1->bandwidthburst,OP_EQ, r1->bandwidthburst);
- tt_int_op(rp1->bandwidthcapacity,OP_EQ, r1->bandwidthcapacity);
- tt_assert(crypto_pk_cmp_keys(rp1->onion_pkey, pk1) == 0);
- tt_assert(crypto_pk_cmp_keys(rp1->identity_pkey, pk2) == 0);
- //tt_assert(rp1->exit_policy == NULL);
- tor_free(buf);
- strlcpy(buf2,
- "router Fred 10.3.2.1 9005 0 0\n"
- "platform Tor "VERSION" on ", sizeof(buf2));
- strlcat(buf2, get_uname(), sizeof(buf2));
- strlcat(buf2, "\n"
- "protocols Link 1 2 Circuit 1\n"
- "published 1970-01-01 00:00:05\n"
- "fingerprint ", sizeof(buf2));
- tt_assert(!crypto_pk_get_fingerprint(pk1, fingerprint, 1));
- strlcat(buf2, fingerprint, sizeof(buf2));
- strlcat(buf2, "\nuptime 0\n"
- "bandwidth 3000 3000 3000\n", sizeof(buf2));
- strlcat(buf2, "onion-key\n", sizeof(buf2));
- strlcat(buf2, pk2_str, sizeof(buf2));
- strlcat(buf2, "signing-key\n", sizeof(buf2));
- strlcat(buf2, pk1_str, sizeof(buf2));
- strlcat(buf2, "hidden-service-dir\n", sizeof(buf2));
- strlcat(buf2, "ntor-onion-key "
- "skyinAnvardNostarsNomoonNowindormistsorsnow=\n", sizeof(buf2));
- strlcat(buf2, "accept *:80\nreject 18.0.0.0/8:24\n", sizeof(buf2));
- strlcat(buf2, "router-signature\n", sizeof(buf2));
- buf = router_dump_router_to_string(r2, pk1);
- buf[strlen(buf2)] = '\0'; /* Don't compare the sig; it's never the same
- * twice */
- tt_str_op(buf,OP_EQ, buf2);
- tor_free(buf);
- buf = router_dump_router_to_string(r2, pk1);
- cp = buf;
- rp2 = router_parse_entry_from_string((const char*)cp,NULL,1,0,NULL,NULL);
- tt_assert(rp2);
- tt_int_op(rp2->addr,OP_EQ, r2->addr);
- tt_int_op(rp2->or_port,OP_EQ, r2->or_port);
- tt_int_op(rp2->dir_port,OP_EQ, r2->dir_port);
- tt_int_op(rp2->bandwidthrate,OP_EQ, r2->bandwidthrate);
- tt_int_op(rp2->bandwidthburst,OP_EQ, r2->bandwidthburst);
- tt_int_op(rp2->bandwidthcapacity,OP_EQ, r2->bandwidthcapacity);
- tt_mem_op(rp2->onion_curve25519_pkey->public_key,OP_EQ,
- r2->onion_curve25519_pkey->public_key,
- CURVE25519_PUBKEY_LEN);
- tt_assert(crypto_pk_cmp_keys(rp2->onion_pkey, pk2) == 0);
- tt_assert(crypto_pk_cmp_keys(rp2->identity_pkey, pk1) == 0);
- tt_int_op(smartlist_len(rp2->exit_policy),OP_EQ, 2);
- p = smartlist_get(rp2->exit_policy, 0);
- tt_int_op(p->policy_type,OP_EQ, ADDR_POLICY_ACCEPT);
- tt_assert(tor_addr_is_null(&p->addr));
- tt_int_op(p->maskbits,OP_EQ, 0);
- tt_int_op(p->prt_min,OP_EQ, 80);
- tt_int_op(p->prt_max,OP_EQ, 80);
- p = smartlist_get(rp2->exit_policy, 1);
- tt_int_op(p->policy_type,OP_EQ, ADDR_POLICY_REJECT);
- tt_assert(tor_addr_eq(&p->addr, &ex2->addr));
- tt_int_op(p->maskbits,OP_EQ, 8);
- tt_int_op(p->prt_min,OP_EQ, 24);
- tt_int_op(p->prt_max,OP_EQ, 24);
- #if 0
- /* Okay, now for the directories. */
- {
- fingerprint_list = smartlist_new();
- crypto_pk_get_fingerprint(pk2, buf, 1);
- add_fingerprint_to_dir(buf, fingerprint_list, 0);
- crypto_pk_get_fingerprint(pk1, buf, 1);
- add_fingerprint_to_dir(buf, fingerprint_list, 0);
- }
- #endif
- dirserv_free_fingerprint_list();
- done:
- if (r1)
- routerinfo_free(r1);
- if (r2)
- routerinfo_free(r2);
- if (rp2)
- routerinfo_free(rp2);
- tor_free(buf);
- tor_free(pk1_str);
- tor_free(pk2_str);
- if (pk1) crypto_pk_free(pk1);
- if (pk2) crypto_pk_free(pk2);
- if (rp1) routerinfo_free(rp1);
- tor_free(dir1); /* XXXX And more !*/
- tor_free(dir2); /* And more !*/
- }
- #include "failing_routerdescs.inc"
- static void
- test_dir_routerparse_bad(void *arg)
- {
- (void) arg;
- int again;
- routerinfo_t *ri = NULL;
- #define CHECK_OK(s) \
- do { \
- routerinfo_free(ri); \
- ri = router_parse_entry_from_string((s), NULL, 0, 0, NULL, NULL); \
- tt_assert(ri); \
- } while (0)
- #define CHECK_FAIL(s, againval) \
- do { \
- routerinfo_free(ri); \
- again = 999; \
- ri = router_parse_entry_from_string((s), NULL, 0, 0, NULL, &again); \
- tt_assert(ri == NULL); \
- tt_int_op(again, OP_EQ, (againval)); \
- } while (0)
- CHECK_OK(EX_RI_MINIMAL);
- CHECK_OK(EX_RI_MAXIMAL);
- /* good annotations prepended */
- routerinfo_free(ri);
- ri = router_parse_entry_from_string(EX_RI_MINIMAL, NULL, 0, 0,
- "@purpose bridge\n", NULL);
- tt_assert(ri != NULL);
- tt_assert(ri->purpose == ROUTER_PURPOSE_BRIDGE);
- routerinfo_free(ri);
- /* bad annotations prepended. */
- ri = router_parse_entry_from_string(EX_RI_MINIMAL,
- NULL, 0, 0, "@purpose\n", NULL);
- tt_assert(ri == NULL);
- /* bad annotations on router. */
- ri = router_parse_entry_from_string("@purpose\nrouter x\n", NULL, 0, 1,
- NULL, NULL);
- tt_assert(ri == NULL);
- /* unwanted annotations on router. */
- ri = router_parse_entry_from_string("@purpose foo\nrouter x\n", NULL, 0, 0,
- NULL, NULL);
- tt_assert(ri == NULL);
- /* No signature. */
- ri = router_parse_entry_from_string("router x\n", NULL, 0, 0,
- NULL, NULL);
- tt_assert(ri == NULL);
- /* Not a router */
- routerinfo_free(ri);
- ri = router_parse_entry_from_string("hello\n", NULL, 0, 0, NULL, NULL);
- tt_assert(ri == NULL);
- CHECK_FAIL(EX_RI_BAD_SIG1, 1);
- CHECK_FAIL(EX_RI_BAD_SIG2, 1);
- CHECK_FAIL(EX_RI_BAD_TOKENS, 0);
- CHECK_FAIL(EX_RI_BAD_PUBLISHED, 0);
- CHECK_FAIL(EX_RI_NEG_BANDWIDTH, 0);
- CHECK_FAIL(EX_RI_BAD_BANDWIDTH, 0);
- CHECK_FAIL(EX_RI_BAD_BANDWIDTH2, 0);
- CHECK_FAIL(EX_RI_BAD_ONIONKEY1, 0);
- CHECK_FAIL(EX_RI_BAD_ONIONKEY2, 0);
- CHECK_FAIL(EX_RI_BAD_PORTS, 0);
- CHECK_FAIL(EX_RI_BAD_IP, 0);
- CHECK_FAIL(EX_RI_BAD_DIRPORT, 0);
- CHECK_FAIL(EX_RI_BAD_NAME2, 0);
- CHECK_FAIL(EX_RI_BAD_UPTIME, 0);
- CHECK_FAIL(EX_RI_BAD_BANDWIDTH3, 0);
- CHECK_FAIL(EX_RI_BAD_NTOR_KEY, 0);
- CHECK_FAIL(EX_RI_BAD_FINGERPRINT, 0);
- CHECK_FAIL(EX_RI_MISMATCHED_FINGERPRINT, 0);
- CHECK_FAIL(EX_RI_BAD_HAS_ACCEPT6, 0);
- CHECK_FAIL(EX_RI_BAD_NO_EXIT_POLICY, 0);
- CHECK_FAIL(EX_RI_BAD_IPV6_EXIT_POLICY, 0);
- CHECK_FAIL(EX_RI_BAD_FAMILY, 0);
- CHECK_FAIL(EX_RI_ZERO_ORPORT, 0);
- /* This is allowed; we just ignore it. */
- CHECK_OK(EX_RI_BAD_EI_DIGEST);
- #undef CHECK_FAIL
- #undef CHECK_OK
- done:
- routerinfo_free(ri);
- }
- #include "example_extrainfo.inc"
- static void
- test_dir_extrainfo_parsing(void *arg)
- {
- (void) arg;
- #define CHECK_OK(s) \
- do { \
- extrainfo_free(ei); \
- ei = extrainfo_parse_entry_from_string((s), NULL, 0, map, NULL); \
- tt_assert(ei); \
- } while (0)
- #define CHECK_FAIL(s, againval) \
- do { \
- extrainfo_free(ei); \
- again = 999; \
- ei = extrainfo_parse_entry_from_string((s), NULL, 0, map, &again); \
- tt_assert(ei == NULL); \
- tt_int_op(again, OP_EQ, (againval)); \
- } while (0)
- #define ADD(name) \
- do { \
- ri = tor_malloc_zero(sizeof(routerinfo_t)); \
- crypto_pk_t *pk = ri->identity_pkey = crypto_pk_new(); \
- tt_assert(! crypto_pk_read_public_key_from_string(pk, \
- name##_KEY, strlen(name##_KEY))); \
- tt_int_op(0,OP_EQ,base16_decode(d, 20, name##_FP, strlen(name##_FP))); \
- digestmap_set((digestmap_t*)map, d, ri); \
- ri = NULL; \
- } while (0)
- routerinfo_t *ri = NULL;
- char d[20];
- struct digest_ri_map_t *map = NULL;
- extrainfo_t *ei = NULL;
- int again;
- CHECK_OK(EX_EI_MINIMAL);
- tt_assert(ei->pending_sig);
- CHECK_OK(EX_EI_MAXIMAL);
- tt_assert(ei->pending_sig);
- map = (struct digest_ri_map_t *)digestmap_new();
- ADD(EX_EI_MINIMAL);
- ADD(EX_EI_MAXIMAL);
- ADD(EX_EI_BAD_FP);
- ADD(EX_EI_BAD_NICKNAME);
- ADD(EX_EI_BAD_TOKENS);
- ADD(EX_EI_BAD_START);
- ADD(EX_EI_BAD_PUBLISHED);
- CHECK_OK(EX_EI_MINIMAL);
- tt_assert(!ei->pending_sig);
- CHECK_OK(EX_EI_MAXIMAL);
- tt_assert(!ei->pending_sig);
- CHECK_FAIL(EX_EI_BAD_SIG1,1);
- CHECK_FAIL(EX_EI_BAD_SIG2,1);
- CHECK_FAIL(EX_EI_BAD_SIG3,1);
- CHECK_FAIL(EX_EI_BAD_FP,0);
- CHECK_FAIL(EX_EI_BAD_NICKNAME,0);
- CHECK_FAIL(EX_EI_BAD_TOKENS,0);
- CHECK_FAIL(EX_EI_BAD_START,0);
- CHECK_FAIL(EX_EI_BAD_PUBLISHED,0);
- #undef CHECK_OK
- #undef CHECK_FAIL
- done:
- routerinfo_free(ri);
- /* XXXX elements should get freed too */
- digestmap_free((digestmap_t*)map, NULL);
- }
- static void
- test_dir_parse_router_list(void *arg)
- {
- (void) arg;
- smartlist_t *invalid = smartlist_new();
- smartlist_t *dest = smartlist_new();
- smartlist_t *chunks = smartlist_new();
- int dest_has_ri = 1;
- char *list = NULL;
- const char *cp;
- digestmap_t *map = NULL;
- char *mem_op_hex_tmp = NULL;
- routerinfo_t *ri = NULL;
- char d[DIGEST_LEN];
- smartlist_add(chunks, tor_strdup(EX_RI_MINIMAL)); // ri 0
- smartlist_add(chunks, tor_strdup(EX_RI_BAD_PORTS)); // bad ri 0
- smartlist_add(chunks, tor_strdup(EX_EI_MAXIMAL)); // ei 0
- smartlist_add(chunks, tor_strdup(EX_EI_BAD_SIG2)); // bad ei --
- smartlist_add(chunks, tor_strdup(EX_EI_BAD_NICKNAME));// bad ei 0
- smartlist_add(chunks, tor_strdup(EX_RI_BAD_SIG1)); // bad ri --
- smartlist_add(chunks, tor_strdup(EX_EI_BAD_PUBLISHED)); // bad ei 1
- smartlist_add(chunks, tor_strdup(EX_RI_MAXIMAL)); // ri 1
- smartlist_add(chunks, tor_strdup(EX_RI_BAD_FAMILY)); // bad ri 1
- smartlist_add(chunks, tor_strdup(EX_EI_MINIMAL)); // ei 1
- list = smartlist_join_strings(chunks, "", 0, NULL);
- /* First, parse the routers. */
- cp = list;
- tt_int_op(0,OP_EQ,
- router_parse_list_from_string(&cp, NULL, dest, SAVED_NOWHERE,
- 0, 0, NULL, invalid));
- tt_int_op(2, OP_EQ, smartlist_len(dest));
- tt_ptr_op(cp, OP_EQ, list + strlen(list));
- routerinfo_t *r = smartlist_get(dest, 0);
- tt_mem_op(r->cache_info.signed_descriptor_body, OP_EQ,
- EX_RI_MINIMAL, strlen(EX_RI_MINIMAL));
- r = smartlist_get(dest, 1);
- tt_mem_op(r->cache_info.signed_descriptor_body, OP_EQ,
- EX_RI_MAXIMAL, strlen(EX_RI_MAXIMAL));
- tt_int_op(2, OP_EQ, smartlist_len(invalid));
- test_memeq_hex(smartlist_get(invalid, 0),
- "ab9eeaa95e7d45740185b4e519c76ead756277a9");
- test_memeq_hex(smartlist_get(invalid, 1),
- "9a651ee03b64325959e8f1b46f2b689b30750b4c");
- /* Now tidy up */
- SMARTLIST_FOREACH(dest, routerinfo_t *, ri, routerinfo_free(ri));
- SMARTLIST_FOREACH(invalid, uint8_t *, d, tor_free(d));
- smartlist_clear(dest);
- smartlist_clear(invalid);
- /* And check extrainfos. */
- dest_has_ri = 0;
- map = (digestmap_t*)router_get_routerlist()->identity_map;
- ADD(EX_EI_MINIMAL);
- ADD(EX_EI_MAXIMAL);
- ADD(EX_EI_BAD_NICKNAME);
- ADD(EX_EI_BAD_PUBLISHED);
- cp = list;
- tt_int_op(0,OP_EQ,
- router_parse_list_from_string(&cp, NULL, dest, SAVED_NOWHERE,
- 1, 0, NULL, invalid));
- tt_int_op(2, OP_EQ, smartlist_len(dest));
- extrainfo_t *e = smartlist_get(dest, 0);
- tt_mem_op(e->cache_info.signed_descriptor_body, OP_EQ,
- EX_EI_MAXIMAL, strlen(EX_EI_MAXIMAL));
- e = smartlist_get(dest, 1);
- tt_mem_op(e->cache_info.signed_descriptor_body, OP_EQ,
- EX_EI_MINIMAL, strlen(EX_EI_MINIMAL));
- tt_int_op(2, OP_EQ, smartlist_len(invalid));
- test_memeq_hex(smartlist_get(invalid, 0),
- "d5df4aa62ee9ffc9543d41150c9864908e0390af");
- test_memeq_hex(smartlist_get(invalid, 1),
- "f61efd2a7f4531f3687a9043e0de90a862ec64ba");
- done:
- tor_free(list);
- if (dest_has_ri)
- SMARTLIST_FOREACH(dest, routerinfo_t *, rt, routerinfo_free(rt));
- else
- SMARTLIST_FOREACH(dest, extrainfo_t *, ei, extrainfo_free(ei));
- smartlist_free(dest);
- SMARTLIST_FOREACH(invalid, uint8_t *, d, tor_free(d));
- smartlist_free(invalid);
- SMARTLIST_FOREACH(chunks, char *, cp, tor_free(cp));
- smartlist_free(chunks);
- routerinfo_free(ri);
- /* XXXX this leaks: */
- if (map) {
- digestmap_free((digestmap_t*)map, NULL);
- router_get_routerlist()->identity_map =
- (struct digest_ri_map_t*)digestmap_new();
- }
- tor_free(mem_op_hex_tmp);
- #undef ADD
- }
- static download_status_t dls_minimal;
- static download_status_t dls_maximal;
- static download_status_t dls_bad_fingerprint;
- static download_status_t dls_bad_sig2;
- static download_status_t dls_bad_ports;
- static download_status_t dls_bad_tokens;
- static int mock_router_get_dl_status_unrecognized = 0;
- static int mock_router_get_dl_status_calls = 0;
- static download_status_t *
- mock_router_get_dl_status(const char *d)
- {
- ++mock_router_get_dl_status_calls;
- char hex[HEX_DIGEST_LEN+1];
- base16_encode(hex, sizeof(hex), d, DIGEST_LEN);
- if (!strcmp(hex, "3E31D19A69EB719C00B02EC60D13356E3F7A3452")) {
- return &dls_minimal;
- } else if (!strcmp(hex, "581D8A368A0FA854ECDBFAB841D88B3F1B004038")) {
- return &dls_maximal;
- } else if (!strcmp(hex, "2578AE227C6116CDE29B3F0E95709B9872DEE5F1")) {
- return &dls_bad_fingerprint;
- } else if (!strcmp(hex, "16D387D3A58F7DB3CF46638F8D0B90C45C7D769C")) {
- return &dls_bad_sig2;
- } else if (!strcmp(hex, "AB9EEAA95E7D45740185B4E519C76EAD756277A9")) {
- return &dls_bad_ports;
- } else if (!strcmp(hex, "A0CC2CEFAD59DBF19F468BFEE60E0868C804B422")) {
- return &dls_bad_tokens;
- } else {
- ++mock_router_get_dl_status_unrecognized;
- return NULL;
- }
- }
- static void
- test_dir_load_routers(void *arg)
- {
- (void) arg;
- smartlist_t *chunks = smartlist_new();
- smartlist_t *wanted = smartlist_new();
- char buf[DIGEST_LEN];
- char *mem_op_hex_tmp = NULL;
- #define ADD(str) \
- do { \
- tt_int_op(0,OP_EQ,router_get_router_hash(str, strlen(str), buf)); \
- smartlist_add(wanted, tor_strdup(hex_str(buf, DIGEST_LEN))); \
- } while (0)
- MOCK(router_get_dl_status_by_descriptor_digest, mock_router_get_dl_status);
- update_approx_time(1412510400);
- smartlist_add(chunks, tor_strdup(EX_RI_MINIMAL));
- smartlist_add(chunks, tor_strdup(EX_RI_BAD_FINGERPRINT));
- smartlist_add(chunks, tor_strdup(EX_RI_BAD_SIG2));
- smartlist_add(chunks, tor_strdup(EX_RI_MAXIMAL));
- smartlist_add(chunks, tor_strdup(EX_RI_BAD_PORTS));
- smartlist_add(chunks, tor_strdup(EX_RI_BAD_TOKENS));
- /* not ADDing MINIMIAL */
- ADD(EX_RI_MAXIMAL);
- ADD(EX_RI_BAD_FINGERPRINT);
- ADD(EX_RI_BAD_SIG2);
- /* Not ADDing BAD_PORTS */
- ADD(EX_RI_BAD_TOKENS);
- char *list = smartlist_join_strings(chunks, "", 0, NULL);
- tt_int_op(1, OP_EQ,
- router_load_routers_from_string(list, NULL, SAVED_IN_JOURNAL,
- wanted, 1, NULL));
- /* The "maximal" router was added. */
- /* "minimal" was not. */
- tt_int_op(smartlist_len(router_get_routerlist()->routers),OP_EQ,1);
- routerinfo_t *r = smartlist_get(router_get_routerlist()->routers, 0);
- test_memeq_hex(r->cache_info.signed_descriptor_digest,
- "581D8A368A0FA854ECDBFAB841D88B3F1B004038");
- tt_int_op(dls_minimal.n_download_failures, OP_EQ, 0);
- tt_int_op(dls_maximal.n_download_failures, OP_EQ, 0);
- /* "Bad fingerprint" and "Bad tokens" should have gotten marked
- * non-retriable. */
- tt_want_int_op(mock_router_get_dl_status_calls, OP_EQ, 2);
- tt_want_int_op(mock_router_get_dl_status_unrecognized, OP_EQ, 0);
- tt_int_op(dls_bad_fingerprint.n_download_failures, OP_EQ, 255);
- tt_int_op(dls_bad_tokens.n_download_failures, OP_EQ, 255);
- /* bad_sig2 and bad ports" are retriable -- one since only the signature
- * was bad, and one because we didn't ask for it. */
- tt_int_op(dls_bad_sig2.n_download_failures, OP_EQ, 0);
- tt_int_op(dls_bad_ports.n_download_failures, OP_EQ, 0);
- /* Wanted still contains "BAD_SIG2" */
- tt_int_op(smartlist_len(wanted), OP_EQ, 1);
- tt_str_op(smartlist_get(wanted, 0), OP_EQ,
- "E0A3753CEFD54128EAB239F294954121DB23D2EF");
- #undef ADD
- done:
- tor_free(mem_op_hex_tmp);
- UNMOCK(router_get_dl_status_by_descriptor_digest);
- SMARTLIST_FOREACH(chunks, char *, cp, tor_free(cp));
- smartlist_free(chunks);
- SMARTLIST_FOREACH(wanted, char *, cp, tor_free(cp));
- smartlist_free(wanted);
- }
- static int mock_get_by_ei_dd_calls = 0;
- static int mock_get_by_ei_dd_unrecognized = 0;
- static signed_descriptor_t sd_ei_minimal;
- static signed_descriptor_t sd_ei_bad_nickname;
- static signed_descriptor_t sd_ei_maximal;
- static signed_descriptor_t sd_ei_bad_tokens;
- static signed_descriptor_t sd_ei_bad_sig2;
- static signed_descriptor_t *
- mock_get_by_ei_desc_digest(const char *d)
- {
- ++mock_get_by_ei_dd_calls;
- char hex[HEX_DIGEST_LEN+1];
- base16_encode(hex, sizeof(hex), d, DIGEST_LEN);
- if (!strcmp(hex, "11E0EDF526950739F7769810FCACAB8C882FAEEE")) {
- return &sd_ei_minimal;
- } else if (!strcmp(hex, "47803B02A0E70E9E8BDA226CB1D74DE354D67DFF")) {
- return &sd_ei_maximal;
- } else if (!strcmp(hex, "D5DF4AA62EE9FFC9543D41150C9864908E0390AF")) {
- return &sd_ei_bad_nickname;
- } else if (!strcmp(hex, "16D387D3A58F7DB3CF46638F8D0B90C45C7D769C")) {
- return &sd_ei_bad_sig2;
- } else if (!strcmp(hex, "9D90F8C42955BBC57D54FB05E54A3F083AF42E8B")) {
- return &sd_ei_bad_tokens;
- } else {
- ++mock_get_by_ei_dd_unrecognized;
- return NULL;
- }
- }
- static smartlist_t *mock_ei_insert_list = NULL;
- static was_router_added_t
- mock_ei_insert(routerlist_t *rl, extrainfo_t *ei)
- {
- (void) rl;
- smartlist_add(mock_ei_insert_list, ei);
- return ROUTER_ADDED_SUCCESSFULLY;
- }
- static void
- test_dir_load_extrainfo(void *arg)
- {
- (void) arg;
- smartlist_t *chunks = smartlist_new();
- smartlist_t *wanted = smartlist_new();
- char buf[DIGEST_LEN];
- char *mem_op_hex_tmp = NULL;
- #define ADD(str) \
- do { \
- tt_int_op(0,OP_EQ,router_get_extrainfo_hash(str, strlen(str), buf)); \
- smartlist_add(wanted, tor_strdup(hex_str(buf, DIGEST_LEN))); \
- } while (0)
- mock_ei_insert_list = smartlist_new();
- MOCK(router_get_by_extrainfo_digest, mock_get_by_ei_desc_digest);
- MOCK(extrainfo_insert, mock_ei_insert);
- smartlist_add(chunks, tor_strdup(EX_EI_MINIMAL));
- smartlist_add(chunks, tor_strdup(EX_EI_BAD_NICKNAME));
- smartlist_add(chunks, tor_strdup(EX_EI_MAXIMAL));
- smartlist_add(chunks, tor_strdup(EX_EI_BAD_PUBLISHED));
- smartlist_add(chunks, tor_strdup(EX_EI_BAD_TOKENS));
- /* not ADDing MINIMIAL */
- ADD(EX_EI_MAXIMAL);
- ADD(EX_EI_BAD_NICKNAME);
- /* Not ADDing BAD_PUBLISHED */
- ADD(EX_EI_BAD_TOKENS);
- ADD(EX_EI_BAD_SIG2);
- char *list = smartlist_join_strings(chunks, "", 0, NULL);
- router_load_extrainfo_from_string(list, NULL, SAVED_IN_JOURNAL, wanted, 1);
- /* The "maximal" router was added. */
- /* "minimal" was also added, even though we didn't ask for it, since
- * that's what we do with extrainfos. */
- tt_int_op(smartlist_len(mock_ei_insert_list),OP_EQ,2);
- extrainfo_t *e = smartlist_get(mock_ei_insert_list, 0);
- test_memeq_hex(e->cache_info.signed_descriptor_digest,
- "11E0EDF526950739F7769810FCACAB8C882FAEEE");
- e = smartlist_get(mock_ei_insert_list, 1);
- test_memeq_hex(e->cache_info.signed_descriptor_digest,
- "47803B02A0E70E9E8BDA226CB1D74DE354D67DFF");
- tt_int_op(dls_minimal.n_download_failures, OP_EQ, 0);
- tt_int_op(dls_maximal.n_download_failures, OP_EQ, 0);
- /* "Bad nickname" and "Bad tokens" should have gotten marked
- * non-retriable. */
- tt_want_int_op(mock_get_by_ei_dd_calls, OP_EQ, 2);
- tt_want_int_op(mock_get_by_ei_dd_unrecognized, OP_EQ, 0);
- tt_int_op(sd_ei_bad_nickname.ei_dl_status.n_download_failures, OP_EQ, 255);
- tt_int_op(sd_ei_bad_tokens.ei_dl_status.n_download_failures, OP_EQ, 255);
- /* bad_ports is retriable -- because we didn't ask for it. */
- tt_int_op(dls_bad_ports.n_download_failures, OP_EQ, 0);
- /* Wanted still contains "BAD_SIG2" */
- tt_int_op(smartlist_len(wanted), OP_EQ, 1);
- tt_str_op(smartlist_get(wanted, 0), OP_EQ,
- "16D387D3A58F7DB3CF46638F8D0B90C45C7D769C");
- #undef ADD
- done:
- tor_free(mem_op_hex_tmp);
- UNMOCK(router_get_by_extrainfo_digest);
- SMARTLIST_FOREACH(chunks, char *, cp, tor_free(cp));
- smartlist_free(chunks);
- SMARTLIST_FOREACH(wanted, char *, cp, tor_free(cp));
- smartlist_free(wanted);
- }
- static void
- test_dir_versions(void *arg)
- {
- tor_version_t ver1;
- /* Try out version parsing functionality */
- (void)arg;
- tt_int_op(0,OP_EQ, tor_version_parse("0.3.4pre2-cvs", &ver1));
- tt_int_op(0,OP_EQ, ver1.major);
- tt_int_op(3,OP_EQ, ver1.minor);
- tt_int_op(4,OP_EQ, ver1.micro);
- tt_int_op(VER_PRE,OP_EQ, ver1.status);
- tt_int_op(2,OP_EQ, ver1.patchlevel);
- tt_int_op(0,OP_EQ, tor_version_parse("0.3.4rc1", &ver1));
- tt_int_op(0,OP_EQ, ver1.major);
- tt_int_op(3,OP_EQ, ver1.minor);
- tt_int_op(4,OP_EQ, ver1.micro);
- tt_int_op(VER_RC,OP_EQ, ver1.status);
- tt_int_op(1,OP_EQ, ver1.patchlevel);
- tt_int_op(0,OP_EQ, tor_version_parse("1.3.4", &ver1));
- tt_int_op(1,OP_EQ, ver1.major);
- tt_int_op(3,OP_EQ, ver1.minor);
- tt_int_op(4,OP_EQ, ver1.micro);
- tt_int_op(VER_RELEASE,OP_EQ, ver1.status);
- tt_int_op(0,OP_EQ, ver1.patchlevel);
- tt_int_op(0,OP_EQ, tor_version_parse("1.3.4.999", &ver1));
- tt_int_op(1,OP_EQ, ver1.major);
- tt_int_op(3,OP_EQ, ver1.minor);
- tt_int_op(4,OP_EQ, ver1.micro);
- tt_int_op(VER_RELEASE,OP_EQ, ver1.status);
- tt_int_op(999,OP_EQ, ver1.patchlevel);
- tt_int_op(0,OP_EQ, tor_version_parse("0.1.2.4-alpha", &ver1));
- tt_int_op(0,OP_EQ, ver1.major);
- tt_int_op(1,OP_EQ, ver1.minor);
- tt_int_op(2,OP_EQ, ver1.micro);
- tt_int_op(4,OP_EQ, ver1.patchlevel);
- tt_int_op(VER_RELEASE,OP_EQ, ver1.status);
- tt_str_op("alpha",OP_EQ, ver1.status_tag);
- tt_int_op(0,OP_EQ, tor_version_parse("0.1.2.4", &ver1));
- tt_int_op(0,OP_EQ, ver1.major);
- tt_int_op(1,OP_EQ, ver1.minor);
- tt_int_op(2,OP_EQ, ver1.micro);
- tt_int_op(4,OP_EQ, ver1.patchlevel);
- tt_int_op(VER_RELEASE,OP_EQ, ver1.status);
- tt_str_op("",OP_EQ, ver1.status_tag);
- #define tt_versionstatus_op(vs1, op, vs2) \
- tt_assert_test_type(vs1,vs2,#vs1" "#op" "#vs2,version_status_t, \
- (val1_ op val2_),"%d",TT_EXIT_TEST_FUNCTION)
- #define test_v_i_o(val, ver, lst) \
- tt_versionstatus_op(val, OP_EQ, tor_version_is_obsolete(ver, lst))
- /* make sure tor_version_is_obsolete() works */
- test_v_i_o(VS_OLD, "0.0.1", "Tor 0.0.2");
- test_v_i_o(VS_OLD, "0.0.1", "0.0.2, Tor 0.0.3");
- test_v_i_o(VS_OLD, "0.0.1", "0.0.2,Tor 0.0.3");
- test_v_i_o(VS_OLD, "0.0.1","0.0.3,BetterTor 0.0.1");
- test_v_i_o(VS_RECOMMENDED, "0.0.2", "Tor 0.0.2,Tor 0.0.3");
- test_v_i_o(VS_NEW_IN_SERIES, "0.0.2", "Tor 0.0.2pre1,Tor 0.0.3");
- test_v_i_o(VS_OLD, "0.0.2", "Tor 0.0.2.1,Tor 0.0.3");
- test_v_i_o(VS_NEW, "0.1.0", "Tor 0.0.2,Tor 0.0.3");
- test_v_i_o(VS_RECOMMENDED, "0.0.7rc2", "0.0.7,Tor 0.0.7rc2,Tor 0.0.8");
- test_v_i_o(VS_OLD, "0.0.5.0", "0.0.5.1-cvs");
- test_v_i_o(VS_NEW_IN_SERIES, "0.0.5.1-cvs", "0.0.5, 0.0.6");
- /* Not on list, but newer than any in same series. */
- test_v_i_o(VS_NEW_IN_SERIES, "0.1.0.3",
- "Tor 0.1.0.2,Tor 0.0.9.5,Tor 0.1.1.0");
- /* Series newer than any on list. */
- test_v_i_o(VS_NEW, "0.1.2.3", "Tor 0.1.0.2,Tor 0.0.9.5,Tor 0.1.1.0");
- /* Series older than any on list. */
- test_v_i_o(VS_OLD, "0.0.1.3", "Tor 0.1.0.2,Tor 0.0.9.5,Tor 0.1.1.0");
- /* Not on list, not newer than any on same series. */
- test_v_i_o(VS_UNRECOMMENDED, "0.1.0.1",
- "Tor 0.1.0.2,Tor 0.0.9.5,Tor 0.1.1.0");
- /* On list, not newer than any on same series. */
- test_v_i_o(VS_UNRECOMMENDED,
- "0.1.0.1", "Tor 0.1.0.2,Tor 0.0.9.5,Tor 0.1.1.0");
- tt_int_op(0,OP_EQ, tor_version_as_new_as("Tor 0.0.5", "0.0.9pre1-cvs"));
- tt_int_op(1,OP_EQ, tor_version_as_new_as(
- "Tor 0.0.8 on Darwin 64-121-192-100.c3-0."
- "sfpo-ubr1.sfrn-sfpo.ca.cable.rcn.com Power Macintosh",
- "0.0.8rc2"));
- tt_int_op(0,OP_EQ, tor_version_as_new_as(
- "Tor 0.0.8 on Darwin 64-121-192-100.c3-0."
- "sfpo-ubr1.sfrn-sfpo.ca.cable.rcn.com Power Macintosh", "0.0.8.2"));
- /* Now try svn revisions. */
- tt_int_op(1,OP_EQ, tor_version_as_new_as("Tor 0.2.1.0-dev (r100)",
- "Tor 0.2.1.0-dev (r99)"));
- tt_int_op(1,OP_EQ, tor_version_as_new_as(
- "Tor 0.2.1.0-dev (r100) on Banana Jr",
- "Tor 0.2.1.0-dev (r99) on Hal 9000"));
- tt_int_op(1,OP_EQ, tor_version_as_new_as("Tor 0.2.1.0-dev (r100)",
- "Tor 0.2.1.0-dev on Colossus"));
- tt_int_op(0,OP_EQ, tor_version_as_new_as("Tor 0.2.1.0-dev (r99)",
- "Tor 0.2.1.0-dev (r100)"));
- tt_int_op(0,OP_EQ, tor_version_as_new_as("Tor 0.2.1.0-dev (r99) on MCP",
- "Tor 0.2.1.0-dev (r100) on AM"));
- tt_int_op(0,OP_EQ, tor_version_as_new_as("Tor 0.2.1.0-dev",
- "Tor 0.2.1.0-dev (r99)"));
- tt_int_op(1,OP_EQ, tor_version_as_new_as("Tor 0.2.1.1",
- "Tor 0.2.1.0-dev (r99)"));
- /* Now try git revisions */
- tt_int_op(0,OP_EQ, tor_version_parse("0.5.6.7 (git-ff00ff)", &ver1));
- tt_int_op(0,OP_EQ, ver1.major);
- tt_int_op(5,OP_EQ, ver1.minor);
- tt_int_op(6,OP_EQ, ver1.micro);
- tt_int_op(7,OP_EQ, ver1.patchlevel);
- tt_int_op(3,OP_EQ, ver1.git_tag_len);
- tt_mem_op(ver1.git_tag,OP_EQ, "\xff\x00\xff", 3);
- tt_int_op(-1,OP_EQ, tor_version_parse("0.5.6.7 (git-ff00xx)", &ver1));
- tt_int_op(-1,OP_EQ, tor_version_parse("0.5.6.7 (git-ff00fff)", &ver1));
- tt_int_op(0,OP_EQ, tor_version_parse("0.5.6.7 (git ff00fff)", &ver1));
- done:
- ;
- }
- /** Run unit tests for directory fp_pair functions. */
- static void
- test_dir_fp_pairs(void *arg)
- {
- smartlist_t *sl = smartlist_new();
- fp_pair_t *pair;
- (void)arg;
- dir_split_resource_into_fingerprint_pairs(
- /* Two pairs, out of order, with one duplicate. */
- "73656372657420646174612E0000000000FFFFFF-"
- "557365204145532d32353620696e73746561642e+"
- "73656372657420646174612E0000000000FFFFFF-"
- "557365204145532d32353620696e73746561642e+"
- "48657861646563696d616c2069736e277420736f-"
- "676f6f6420666f7220686964696e6720796f7572.z", sl);
- tt_int_op(smartlist_len(sl),OP_EQ, 2);
- pair = smartlist_get(sl, 0);
- tt_mem_op(pair->first,OP_EQ, "Hexadecimal isn't so", DIGEST_LEN);
- tt_mem_op(pair->second,OP_EQ, "good for hiding your", DIGEST_LEN);
- pair = smartlist_get(sl, 1);
- tt_mem_op(pair->first,OP_EQ, "secret data.\0\0\0\0\0\xff\xff\xff",
- DIGEST_LEN);
- tt_mem_op(pair->second,OP_EQ, "Use AES-256 instead.", DIGEST_LEN);
- done:
- SMARTLIST_FOREACH(sl, fp_pair_t *, pair, tor_free(pair));
- smartlist_free(sl);
- }
- static void
- test_dir_split_fps(void *testdata)
- {
- smartlist_t *sl = smartlist_new();
- char *mem_op_hex_tmp = NULL;
- (void)testdata;
- /* Some example hex fingerprints and their base64 equivalents */
- #define HEX1 "Fe0daff89127389bc67558691231234551193EEE"
- #define HEX2 "Deadbeef99999991111119999911111111f00ba4"
- #define HEX3 "b33ff00db33ff00db33ff00db33ff00db33ff00d"
- #define HEX256_1 \
- "f3f3f3f3fbbbbf3f3f3f3fbbbf3f3f3f3fbbbbf3f3f3f3fbbbf3f3f3f3fbbbbf"
- #define HEX256_2 \
- "cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccCCc"
- #define HEX256_3 \
- "0123456789ABCdef0123456789ABCdef0123456789ABCdef0123456789ABCdef"
- #define B64_1 "/g2v+JEnOJvGdVhpEjEjRVEZPu4"
- #define B64_2 "3q2+75mZmZERERmZmRERERHwC6Q"
- #define B64_256_1 "8/Pz8/u7vz8/Pz+7vz8/Pz+7u/Pz8/P7u/Pz8/P7u78"
- #define B64_256_2 "zMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMw"
- /* no flags set */
- dir_split_resource_into_fingerprints("A+C+B", sl, NULL, 0);
- tt_int_op(smartlist_len(sl), OP_EQ, 3);
- tt_str_op(smartlist_get(sl, 0), OP_EQ, "A");
- tt_str_op(smartlist_get(sl, 1), OP_EQ, "C");
- tt_str_op(smartlist_get(sl, 2), OP_EQ, "B");
- SMARTLIST_FOREACH(sl, char *, cp, tor_free(cp));
- smartlist_clear(sl);
- /* uniq strings. */
- dir_split_resource_into_fingerprints("A+C+B+A+B+B", sl, NULL, DSR_SORT_UNIQ);
- tt_int_op(smartlist_len(sl), OP_EQ, 3);
- tt_str_op(smartlist_get(sl, 0), OP_EQ, "A");
- tt_str_op(smartlist_get(sl, 1), OP_EQ, "B");
- tt_str_op(smartlist_get(sl, 2), OP_EQ, "C");
- SMARTLIST_FOREACH(sl, char *, cp, tor_free(cp));
- smartlist_clear(sl);
- /* Decode hex. */
- dir_split_resource_into_fingerprints(HEX1"+"HEX2, sl, NULL, DSR_HEX);
- tt_int_op(smartlist_len(sl), OP_EQ, 2);
- test_mem_op_hex(smartlist_get(sl, 0), OP_EQ, HEX1);
- test_mem_op_hex(smartlist_get(sl, 1), OP_EQ, HEX2);
- SMARTLIST_FOREACH(sl, char *, cp, tor_free(cp));
- smartlist_clear(sl);
- /* decode hex and drop weirdness. */
- dir_split_resource_into_fingerprints(HEX1"+bogus+"HEX2"+"HEX256_1,
- sl, NULL, DSR_HEX);
- tt_int_op(smartlist_len(sl), OP_EQ, 2);
- test_mem_op_hex(smartlist_get(sl, 0), OP_EQ, HEX1);
- test_mem_op_hex(smartlist_get(sl, 1), OP_EQ, HEX2);
- SMARTLIST_FOREACH(sl, char *, cp, tor_free(cp));
- smartlist_clear(sl);
- /* Decode long hex */
- dir_split_resource_into_fingerprints(HEX256_1"+"HEX256_2"+"HEX2"+"HEX256_3,
- sl, NULL, DSR_HEX|DSR_DIGEST256);
- tt_int_op(smartlist_len(sl), OP_EQ, 3);
- test_mem_op_hex(smartlist_get(sl, 0), OP_EQ, HEX256_1);
- test_mem_op_hex(smartlist_get(sl, 1), OP_EQ, HEX256_2);
- test_mem_op_hex(smartlist_get(sl, 2), OP_EQ, HEX256_3);
- SMARTLIST_FOREACH(sl, char *, cp, tor_free(cp));
- smartlist_clear(sl);
- /* Decode hex and sort. */
- dir_split_resource_into_fingerprints(HEX1"+"HEX2"+"HEX3"+"HEX2,
- sl, NULL, DSR_HEX|DSR_SORT_UNIQ);
- tt_int_op(smartlist_len(sl), OP_EQ, 3);
- test_mem_op_hex(smartlist_get(sl, 0), OP_EQ, HEX3);
- test_mem_op_hex(smartlist_get(sl, 1), OP_EQ, HEX2);
- test_mem_op_hex(smartlist_get(sl, 2), OP_EQ, HEX1);
- SMARTLIST_FOREACH(sl, char *, cp, tor_free(cp));
- smartlist_clear(sl);
- /* Decode long hex and sort */
- dir_split_resource_into_fingerprints(HEX256_1"+"HEX256_2"+"HEX256_3
- "+"HEX256_1,
- sl, NULL,
- DSR_HEX|DSR_DIGEST256|DSR_SORT_UNIQ);
- tt_int_op(smartlist_len(sl), OP_EQ, 3);
- test_mem_op_hex(smartlist_get(sl, 0), OP_EQ, HEX256_3);
- test_mem_op_hex(smartlist_get(sl, 1), OP_EQ, HEX256_2);
- test_mem_op_hex(smartlist_get(sl, 2), OP_EQ, HEX256_1);
- SMARTLIST_FOREACH(sl, char *, cp, tor_free(cp));
- smartlist_clear(sl);
- /* Decode base64 */
- dir_split_resource_into_fingerprints(B64_1"-"B64_2, sl, NULL, DSR_BASE64);
- tt_int_op(smartlist_len(sl), OP_EQ, 2);
- test_mem_op_hex(smartlist_get(sl, 0), OP_EQ, HEX1);
- test_mem_op_hex(smartlist_get(sl, 1), OP_EQ, HEX2);
- SMARTLIST_FOREACH(sl, char *, cp, tor_free(cp));
- smartlist_clear(sl);
- /* Decode long base64 */
- dir_split_resource_into_fingerprints(B64_256_1"-"B64_256_2,
- sl, NULL, DSR_BASE64|DSR_DIGEST256);
- tt_int_op(smartlist_len(sl), OP_EQ, 2);
- test_mem_op_hex(smartlist_get(sl, 0), OP_EQ, HEX256_1);
- test_mem_op_hex(smartlist_get(sl, 1), OP_EQ, HEX256_2);
- SMARTLIST_FOREACH(sl, char *, cp, tor_free(cp));
- smartlist_clear(sl);
- dir_split_resource_into_fingerprints(B64_256_1,
- sl, NULL, DSR_BASE64|DSR_DIGEST256);
- tt_int_op(smartlist_len(sl), OP_EQ, 1);
- test_mem_op_hex(smartlist_get(sl, 0), OP_EQ, HEX256_1);
- SMARTLIST_FOREACH(sl, char *, cp, tor_free(cp));
- smartlist_clear(sl);
- done:
- SMARTLIST_FOREACH(sl, char *, cp, tor_free(cp));
- smartlist_free(sl);
- tor_free(mem_op_hex_tmp);
- }
- static void
- test_dir_measured_bw_kb(void *arg)
- {
- measured_bw_line_t mbwl;
- int i;
- const char *lines_pass[] = {
- "node_id=$557365204145532d32353620696e73746561642e bw=1024\n",
- "node_id=$557365204145532d32353620696e73746561642e\t bw=1024 \n",
- " node_id=$557365204145532d32353620696e73746561642e bw=1024\n",
- "\tnoise\tnode_id=$557365204145532d32353620696e73746561642e "
- "bw=1024 junk=007\n",
- "misc=junk node_id=$557365204145532d32353620696e73746561642e "
- "bw=1024 junk=007\n",
- "end"
- };
- const char *lines_fail[] = {
- /* Test possible python stupidity on input */
- "node_id=None bw=1024\n",
- "node_id=$None bw=1024\n",
- "node_id=$557365204145532d32353620696e73746561642e bw=None\n",
- "node_id=$557365204145532d32353620696e73746561642e bw=1024.0\n",
- "node_id=$557365204145532d32353620696e73746561642e bw=.1024\n",
- "node_id=$557365204145532d32353620696e73746561642e bw=1.024\n",
- "node_id=$557365204145532d32353620696e73746561642e bw=1024 bw=0\n",
- "node_id=$557365204145532d32353620696e73746561642e bw=1024 bw=None\n",
- "node_id=$557365204145532d32353620696e73746561642e bw=-1024\n",
- /* Test incomplete writes due to race conditions, partial copies, etc */
- "node_i",
- "node_i\n",
- "node_id=",
- "node_id=\n",
- "node_id=$557365204145532d32353620696e73746561642e bw=",
- "node_id=$557365204145532d32353620696e73746561642e bw=1024",
- "node_id=$557365204145532d32353620696e73746561642e bw=\n",
- "node_id=$557365204145532d32353620696e7374",
- "node_id=$557365204145532d32353620696e7374\n",
- "",
- "\n",
- " \n ",
- " \n\n",
- /* Test assorted noise */
- " node_id= ",
- "node_id==$557365204145532d32353620696e73746561642e bw==1024\n",
- "node_id=$55736520414552d32353620696e73746561642e bw=1024\n",
- "node_id=557365204145532d32353620696e73746561642e bw=1024\n",
- "node_id= $557365204145532d32353620696e73746561642e bw=0.23\n",
- "end"
- };
- (void)arg;
- for (i = 0; strcmp(lines_fail[i], "end"); i++) {
- //fprintf(stderr, "Testing: %s\n", lines_fail[i]);
- tt_assert(measured_bw_line_parse(&mbwl, lines_fail[i]) == -1);
- }
- for (i = 0; strcmp(lines_pass[i], "end"); i++) {
- //fprintf(stderr, "Testing: %s %d\n", lines_pass[i], TOR_ISSPACE('\n'));
- tt_assert(measured_bw_line_parse(&mbwl, lines_pass[i]) == 0);
- tt_assert(mbwl.bw_kb == 1024);
- tt_assert(strcmp(mbwl.node_hex,
- "557365204145532d32353620696e73746561642e") == 0);
- }
- done:
- return;
- }
- #define MBWC_INIT_TIME 1000
- /** Do the measured bandwidth cache unit test */
- static void
- test_dir_measured_bw_kb_cache(void *arg)
- {
- /* Initial fake time_t for testing */
- time_t curr = MBWC_INIT_TIME;
- /* Some measured_bw_line_ts */
- measured_bw_line_t mbwl[3];
- /* For receiving output on cache queries */
- long bw;
- time_t as_of;
- /* First, clear the cache and assert that it's empty */
- (void)arg;
- dirserv_clear_measured_bw_cache();
- tt_int_op(dirserv_get_measured_bw_cache_size(),OP_EQ, 0);
- /*
- * Set up test mbwls; none of the dirserv_cache_*() functions care about
- * the node_hex field.
- */
- memset(mbwl[0].node_id, 0x01, DIGEST_LEN);
- mbwl[0].bw_kb = 20;
- memset(mbwl[1].node_id, 0x02, DIGEST_LEN);
- mbwl[1].bw_kb = 40;
- memset(mbwl[2].node_id, 0x03, DIGEST_LEN);
- mbwl[2].bw_kb = 80;
- /* Try caching something */
- dirserv_cache_measured_bw(&(mbwl[0]), curr);
- tt_int_op(dirserv_get_measured_bw_cache_size(),OP_EQ, 1);
- /* Okay, let's see if we can retrieve it */
- tt_assert(dirserv_query_measured_bw_cache_kb(mbwl[0].node_id,&bw, &as_of));
- tt_int_op(bw,OP_EQ, 20);
- tt_int_op(as_of,OP_EQ, MBWC_INIT_TIME);
- /* Try retrieving it without some outputs */
- tt_assert(dirserv_query_measured_bw_cache_kb(mbwl[0].node_id,NULL, NULL));
- tt_assert(dirserv_query_measured_bw_cache_kb(mbwl[0].node_id,&bw, NULL));
- tt_int_op(bw,OP_EQ, 20);
- tt_assert(dirserv_query_measured_bw_cache_kb(mbwl[0].node_id,NULL,&as_of));
- tt_int_op(as_of,OP_EQ, MBWC_INIT_TIME);
- /* Now expire it */
- curr += MAX_MEASUREMENT_AGE + 1;
- dirserv_expire_measured_bw_cache(curr);
- /* Check that the cache is empty */
- tt_int_op(dirserv_get_measured_bw_cache_size(),OP_EQ, 0);
- /* Check that we can't retrieve it */
- tt_assert(!dirserv_query_measured_bw_cache_kb(mbwl[0].node_id, NULL,NULL));
- /* Try caching a few things now */
- dirserv_cache_measured_bw(&(mbwl[0]), curr);
- tt_int_op(dirserv_get_measured_bw_cache_size(),OP_EQ, 1);
- curr += MAX_MEASUREMENT_AGE / 4;
- dirserv_cache_measured_bw(&(mbwl[1]), curr);
- tt_int_op(dirserv_get_measured_bw_cache_size(),OP_EQ, 2);
- curr += MAX_MEASUREMENT_AGE / 4;
- dirserv_cache_measured_bw(&(mbwl[2]), curr);
- tt_int_op(dirserv_get_measured_bw_cache_size(),OP_EQ, 3);
- curr += MAX_MEASUREMENT_AGE / 4 + 1;
- /* Do an expire that's too soon to get any of them */
- dirserv_expire_measured_bw_cache(curr);
- tt_int_op(dirserv_get_measured_bw_cache_size(),OP_EQ, 3);
- /* Push the oldest one off the cliff */
- curr += MAX_MEASUREMENT_AGE / 4;
- dirserv_expire_measured_bw_cache(curr);
- tt_int_op(dirserv_get_measured_bw_cache_size(),OP_EQ, 2);
- /* And another... */
- curr += MAX_MEASUREMENT_AGE / 4;
- dirserv_expire_measured_bw_cache(curr);
- tt_int_op(dirserv_get_measured_bw_cache_size(),OP_EQ, 1);
- /* This should empty it out again */
- curr += MAX_MEASUREMENT_AGE / 4;
- dirserv_expire_measured_bw_cache(curr);
- tt_int_op(dirserv_get_measured_bw_cache_size(),OP_EQ, 0);
- done:
- return;
- }
- static void
- test_dir_param_voting(void *arg)
- {
- networkstatus_t vote1, vote2, vote3, vote4;
- smartlist_t *votes = smartlist_new();
- char *res = NULL;
- /* dirvote_compute_params only looks at the net_params field of the votes,
- so that's all we need to set.
- */
- (void)arg;
- memset(&vote1, 0, sizeof(vote1));
- memset(&vote2, 0, sizeof(vote2));
- memset(&vote3, 0, sizeof(vote3));
- memset(&vote4, 0, sizeof(vote4));
- vote1.net_params = smartlist_new();
- vote2.net_params = smartlist_new();
- vote3.net_params = smartlist_new();
- vote4.net_params = smartlist_new();
- smartlist_split_string(vote1.net_params,
- "ab=90 abcd=20 cw=50 x-yz=-99", NULL, 0, 0);
- smartlist_split_string(vote2.net_params,
- "ab=27 cw=5 x-yz=88", NULL, 0, 0);
- smartlist_split_string(vote3.net_params,
- "abcd=20 c=60 cw=500 x-yz=-9 zzzzz=101", NULL, 0, 0);
- smartlist_split_string(vote4.net_params,
- "ab=900 abcd=200 c=1 cw=51 x-yz=100", NULL, 0, 0);
- tt_int_op(100,OP_EQ, networkstatus_get_param(&vote4, "x-yz", 50, 0, 300));
- tt_int_op(222,OP_EQ, networkstatus_get_param(&vote4, "foobar", 222, 0, 300));
- tt_int_op(80,OP_EQ, networkstatus_get_param(&vote4, "ab", 12, 0, 80));
- tt_int_op(-8,OP_EQ, networkstatus_get_param(&vote4, "ab", -12, -100, -8));
- tt_int_op(0,OP_EQ, networkstatus_get_param(&vote4, "foobar", 0, -100, 8));
- smartlist_add(votes, &vote1);
- /* Do the first tests without adding all the other votes, for
- * networks without many dirauths. */
- res = dirvote_compute_params(votes, 12, 2);
- tt_str_op(res,OP_EQ, "");
- tor_free(res);
- res = dirvote_compute_params(votes, 12, 1);
- tt_str_op(res,OP_EQ, "ab=90 abcd=20 cw=50 x-yz=-99");
- tor_free(res);
- smartlist_add(votes, &vote2);
- res = dirvote_compute_params(votes, 12, 2);
- tt_str_op(res,OP_EQ, "ab=27 cw=5 x-yz=-99");
- tor_free(res);
- res = dirvote_compute_params(votes, 12, 3);
- tt_str_op(res,OP_EQ, "ab=27 cw=5 x-yz=-99");
- tor_free(res);
- res = dirvote_compute_params(votes, 12, 6);
- tt_str_op(res,OP_EQ, "");
- tor_free(res);
- smartlist_add(votes, &vote3);
- res = dirvote_compute_params(votes, 12, 3);
- tt_str_op(res,OP_EQ, "ab=27 abcd=20 cw=50 x-yz=-9");
- tor_free(res);
- res = dirvote_compute_params(votes, 12, 5);
- tt_str_op(res,OP_EQ, "cw=50 x-yz=-9");
- tor_free(res);
- res = dirvote_compute_params(votes, 12, 9);
- tt_str_op(res,OP_EQ, "cw=50 x-yz=-9");
- tor_free(res);
- smartlist_add(votes, &vote4);
- res = dirvote_compute_params(votes, 12, 4);
- tt_str_op(res,OP_EQ, "ab=90 abcd=20 cw=50 x-yz=-9");
- tor_free(res);
- res = dirvote_compute_params(votes, 12, 5);
- tt_str_op(res,OP_EQ, "ab=90 abcd=20 cw=50 x-yz=-9");
- tor_free(res);
- /* Test that the special-cased "at least three dirauths voted for
- * this param" logic works as expected. */
- res = dirvote_compute_params(votes, 12, 6);
- tt_str_op(res,OP_EQ, "ab=90 abcd=20 cw=50 x-yz=-9");
- tor_free(res);
- res = dirvote_compute_params(votes, 12, 10);
- tt_str_op(res,OP_EQ, "ab=90 abcd=20 cw=50 x-yz=-9");
- tor_free(res);
- done:
- tor_free(res);
- SMARTLIST_FOREACH(vote1.net_params, char *, cp, tor_free(cp));
- SMARTLIST_FOREACH(vote2.net_params, char *, cp, tor_free(cp));
- SMARTLIST_FOREACH(vote3.net_params, char *, cp, tor_free(cp));
- SMARTLIST_FOREACH(vote4.net_params, char *, cp, tor_free(cp));
- smartlist_free(vote1.net_params);
- smartlist_free(vote2.net_params);
- smartlist_free(vote3.net_params);
- smartlist_free(vote4.net_params);
- smartlist_free(votes);
- return;
- }
- extern const char AUTHORITY_CERT_1[];
- extern const char AUTHORITY_SIGNKEY_1[];
- extern const char AUTHORITY_CERT_2[];
- extern const char AUTHORITY_SIGNKEY_2[];
- extern const char AUTHORITY_CERT_3[];
- extern const char AUTHORITY_SIGNKEY_3[];
- /** Helper: Test that two networkstatus_voter_info_t do in fact represent the
- * same voting authority, and that they do in fact have all the same
- * information. */
- static void
- test_same_voter(networkstatus_voter_info_t *v1,
- networkstatus_voter_info_t *v2)
- {
- tt_str_op(v1->nickname,OP_EQ, v2->nickname);
- tt_mem_op(v1->identity_digest,OP_EQ, v2->identity_digest, DIGEST_LEN);
- tt_str_op(v1->address,OP_EQ, v2->address);
- tt_int_op(v1->addr,OP_EQ, v2->addr);
- tt_int_op(v1->dir_port,OP_EQ, v2->dir_port);
- tt_int_op(v1->or_port,OP_EQ, v2->or_port);
- tt_str_op(v1->contact,OP_EQ, v2->contact);
- tt_mem_op(v1->vote_digest,OP_EQ, v2->vote_digest, DIGEST_LEN);
- done:
- ;
- }
- /** Helper: Make a new routerinfo containing the right information for a
- * given vote_routerstatus_t. */
- static routerinfo_t *
- generate_ri_from_rs(const vote_routerstatus_t *vrs)
- {
- routerinfo_t *r;
- const routerstatus_t *rs = &vrs->status;
- static time_t published = 0;
- r = tor_malloc_zero(sizeof(routerinfo_t));
- memcpy(r->cache_info.identity_digest, rs->identity_digest, DIGEST_LEN);
- memcpy(r->cache_info.signed_descriptor_digest, rs->descriptor_digest,
- DIGEST_LEN);
- r->cache_info.do_not_cache = 1;
- r->cache_info.routerlist_index = -1;
- r->cache_info.signed_descriptor_body =
- tor_strdup("123456789012345678901234567890123");
- r->cache_info.signed_descriptor_len =
- strlen(r->cache_info.signed_descriptor_body);
- r->exit_policy = smartlist_new();
- r->cache_info.published_on = ++published + time(NULL);
- if (rs->has_bandwidth) {
- /*
- * Multiply by 1000 because the routerinfo_t and the routerstatus_t
- * seem to use different units (*sigh*) and because we seem stuck on
- * icky and perverse decimal kilobytes (*double sigh*) - see
- * router_get_advertised_bandwidth_capped() of routerlist.c and
- * routerstatus_format_entry() of dirserv.c.
- */
- r->bandwidthrate = rs->bandwidth_kb * 1000;
- r->bandwidthcapacity = rs->bandwidth_kb * 1000;
- }
- return r;
- }
- /** Helper: get a detached signatures document for one or two
- * consensuses. */
- static char *
- get_detached_sigs(networkstatus_t *ns, networkstatus_t *ns2)
- {
- char *r;
- smartlist_t *sl;
- tor_assert(ns && ns->flavor == FLAV_NS);
- sl = smartlist_new();
- smartlist_add(sl,ns);
- if (ns2)
- smartlist_add(sl,ns2);
- r = networkstatus_get_detached_signatures(sl);
- smartlist_free(sl);
- return r;
- }
- /**
- * Generate a routerstatus for v3_networkstatus test
- */
- static vote_routerstatus_t *
- gen_routerstatus_for_v3ns(int idx, time_t now)
- {
- vote_routerstatus_t *vrs=NULL;
- routerstatus_t *rs;
- tor_addr_t addr_ipv6;
- switch (idx) {
- case 0:
- /* Generate the first routerstatus. */
- vrs = tor_malloc_zero(sizeof(vote_routerstatus_t));
- rs = &vrs->status;
- vrs->version = tor_strdup("0.1.2.14");
- rs->published_on = now-1500;
- strlcpy(rs->nickname, "router2", sizeof(rs->nickname));
- memset(rs->identity_digest, 3, DIGEST_LEN);
- memset(rs->descriptor_digest, 78, DIGEST_LEN);
- rs->addr = 0x99008801;
- rs->or_port = 443;
- rs->dir_port = 8000;
- /* all flags but running cleared */
- rs->is_flagged_running = 1;
- break;
- case 1:
- /* Generate the second routerstatus. */
- vrs = tor_malloc_zero(sizeof(vote_routerstatus_t));
- rs = &vrs->status;
- vrs->version = tor_strdup("0.2.0.5");
- rs->published_on = now-1000;
- strlcpy(rs->nickname, "router1", sizeof(rs->nickname));
- memset(rs->identity_digest, 5, DIGEST_LEN);
- memset(rs->descriptor_digest, 77, DIGEST_LEN);
- rs->addr = 0x99009901;
- rs->or_port = 443;
- rs->dir_port = 0;
- tor_addr_parse(&addr_ipv6, "[1:2:3::4]");
- tor_addr_copy(&rs->ipv6_addr, &addr_ipv6);
- rs->ipv6_orport = 4711;
- rs->is_exit = rs->is_stable = rs->is_fast = rs->is_flagged_running =
- rs->is_valid = rs->is_possible_guard = 1;
- break;
- case 2:
- /* Generate the third routerstatus. */
- vrs = tor_malloc_zero(sizeof(vote_routerstatus_t));
- rs = &vrs->status;
- vrs->version = tor_strdup("0.1.0.3");
- rs->published_on = now-1000;
- strlcpy(rs->nickname, "router3", sizeof(rs->nickname));
- memset(rs->identity_digest, 33, DIGEST_LEN);
- memset(rs->descriptor_digest, 79, DIGEST_LEN);
- rs->addr = 0xAA009901;
- rs->or_port = 400;
- rs->dir_port = 9999;
- rs->is_authority = rs->is_exit = rs->is_stable = rs->is_fast =
- rs->is_flagged_running = rs->is_valid =
- rs->is_possible_guard = 1;
- break;
- case 3:
- /* Generate a fourth routerstatus that is not running. */
- vrs = tor_malloc_zero(sizeof(vote_routerstatus_t));
- rs = &vrs->status;
- vrs->version = tor_strdup("0.1.6.3");
- rs->published_on = now-1000;
- strlcpy(rs->nickname, "router4", sizeof(rs->nickname));
- memset(rs->identity_digest, 34, DIGEST_LEN);
- memset(rs->descriptor_digest, 47, DIGEST_LEN);
- rs->addr = 0xC0000203;
- rs->or_port = 500;
- rs->dir_port = 1999;
- /* Running flag (and others) cleared */
- break;
- case 4:
- /* No more for this test; return NULL */
- vrs = NULL;
- break;
- default:
- /* Shouldn't happen */
- tt_assert(0);
- }
- if (vrs) {
- vrs->microdesc = tor_malloc_zero(sizeof(vote_microdesc_hash_t));
- tor_asprintf(&vrs->microdesc->microdesc_hash_line,
- "m 9,10,11,12,13,14,15,16,17 "
- "sha256=xyzajkldsdsajdadlsdjaslsdksdjlsdjsdaskdaaa%d\n",
- idx);
- }
- done:
- return vrs;
- }
- /** Apply tweaks to the vote list for each voter */
- static int
- vote_tweaks_for_v3ns(networkstatus_t *v, int voter, time_t now)
- {
- vote_routerstatus_t *vrs;
- const char *msg = NULL;
- tt_assert(v);
- (void)now;
- if (voter == 1) {
- measured_bw_line_t mbw;
- memset(mbw.node_id, 33, sizeof(mbw.node_id));
- mbw.bw_kb = 1024;
- tt_assert(measured_bw_line_apply(&mbw,
- v->routerstatus_list) == 1);
- } else if (voter == 2 || voter == 3) {
- /* Monkey around with the list a bit */
- vrs = smartlist_get(v->routerstatus_list, 2);
- smartlist_del_keeporder(v->routerstatus_list, 2);
- vote_routerstatus_free(vrs);
- vrs = smartlist_get(v->routerstatus_list, 0);
- vrs->status.is_fast = 1;
- if (voter == 3) {
- vrs = smartlist_get(v->routerstatus_list, 0);
- smartlist_del_keeporder(v->routerstatus_list, 0);
- vote_routerstatus_free(vrs);
- vrs = smartlist_get(v->routerstatus_list, 0);
- memset(vrs->status.descriptor_digest, (int)'Z', DIGEST_LEN);
- tt_assert(router_add_to_routerlist(
- generate_ri_from_rs(vrs), &msg,0,0) >= 0);
- }
- }
- done:
- return 0;
- }
- /**
- * Test a parsed vote_routerstatus_t for v3_networkstatus test
- */
- static void
- test_vrs_for_v3ns(vote_routerstatus_t *vrs, int voter, time_t now)
- {
- routerstatus_t *rs;
- tor_addr_t addr_ipv6;
- tt_assert(vrs);
- rs = &(vrs->status);
- tt_assert(rs);
- /* Split out by digests to test */
- if (tor_memeq(rs->identity_digest,
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3"
- "\x3\x3\x3\x3",
- DIGEST_LEN) &&
- (voter == 1)) {
- /* Check the first routerstatus. */
- tt_str_op(vrs->version,OP_EQ, "0.1.2.14");
- tt_int_op(rs->published_on,OP_EQ, now-1500);
- tt_str_op(rs->nickname,OP_EQ, "router2");
- tt_mem_op(rs->identity_digest,OP_EQ,
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3"
- "\x3\x3\x3\x3",
- DIGEST_LEN);
- tt_mem_op(rs->descriptor_digest,OP_EQ, "NNNNNNNNNNNNNNNNNNNN", DIGEST_LEN);
- tt_int_op(rs->addr,OP_EQ, 0x99008801);
- tt_int_op(rs->or_port,OP_EQ, 443);
- tt_int_op(rs->dir_port,OP_EQ, 8000);
- /* no flags except "running" (16) and "v2dir" (64) */
- tt_u64_op(vrs->flags, OP_EQ, U64_LITERAL(80));
- } else if (tor_memeq(rs->identity_digest,
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5"
- "\x5\x5\x5\x5",
- DIGEST_LEN) &&
- (voter == 1 || voter == 2)) {
- tt_mem_op(rs->identity_digest,OP_EQ,
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5"
- "\x5\x5\x5\x5",
- DIGEST_LEN);
- if (voter == 1) {
- /* Check the second routerstatus. */
- tt_str_op(vrs->version,OP_EQ, "0.2.0.5");
- tt_int_op(rs->published_on,OP_EQ, now-1000);
- tt_str_op(rs->nickname,OP_EQ, "router1");
- }
- tt_mem_op(rs->descriptor_digest,OP_EQ, "MMMMMMMMMMMMMMMMMMMM", DIGEST_LEN);
- tt_int_op(rs->addr,OP_EQ, 0x99009901);
- tt_int_op(rs->or_port,OP_EQ, 443);
- tt_int_op(rs->dir_port,OP_EQ, 0);
- tor_addr_parse(&addr_ipv6, "[1:2:3::4]");
- tt_assert(tor_addr_eq(&rs->ipv6_addr, &addr_ipv6));
- tt_int_op(rs->ipv6_orport,OP_EQ, 4711);
- if (voter == 1) {
- /* all except "authority" (1) and "v2dir" (64) */
- tt_u64_op(vrs->flags, OP_EQ, U64_LITERAL(190));
- } else {
- /* 1023 - authority(1) - madeofcheese(16) - madeoftin(32) - v2dir(256) */
- tt_u64_op(vrs->flags, OP_EQ, U64_LITERAL(718));
- }
- } else if (tor_memeq(rs->identity_digest,
- "\x33\x33\x33\x33\x33\x33\x33\x33\x33\x33"
- "\x33\x33\x33\x33\x33\x33\x33\x33\x33\x33",
- DIGEST_LEN) &&
- (voter == 1 || voter == 2)) {
- /* Check the measured bandwidth bits */
- tt_assert(vrs->has_measured_bw &&
- vrs->measured_bw_kb == 1024);
- } else {
- /*
- * Didn't expect this, but the old unit test only checked some of them,
- * so don't assert.
- */
- /* tt_assert(0); */
- }
- done:
- return;
- }
- /**
- * Test a consensus for v3_networkstatus_test
- */
- static void
- test_consensus_for_v3ns(networkstatus_t *con, time_t now)
- {
- (void)now;
- tt_assert(con);
- tt_assert(!con->cert);
- tt_int_op(2,OP_EQ, smartlist_len(con->routerstatus_list));
- /* There should be two listed routers: one with identity 3, one with
- * identity 5. */
- done:
- return;
- }
- /**
- * Test a router list entry for v3_networkstatus test
- */
- static void
- test_routerstatus_for_v3ns(routerstatus_t *rs, time_t now)
- {
- tor_addr_t addr_ipv6;
- tt_assert(rs);
- /* There should be two listed routers: one with identity 3, one with
- * identity 5. */
- /* This one showed up in 2 digests. */
- if (tor_memeq(rs->identity_digest,
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3"
- "\x3\x3",
- DIGEST_LEN)) {
- tt_mem_op(rs->identity_digest,OP_EQ,
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3",
- DIGEST_LEN);
- tt_mem_op(rs->descriptor_digest,OP_EQ, "NNNNNNNNNNNNNNNNNNNN", DIGEST_LEN);
- tt_assert(!rs->is_authority);
- tt_assert(!rs->is_exit);
- tt_assert(!rs->is_fast);
- tt_assert(!rs->is_possible_guard);
- tt_assert(!rs->is_stable);
- /* (If it wasn't running it wouldn't be here) */
- tt_assert(rs->is_flagged_running);
- tt_assert(!rs->is_valid);
- tt_assert(!rs->is_named);
- /* XXXX check version */
- } else if (tor_memeq(rs->identity_digest,
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5"
- "\x5\x5\x5\x5",
- DIGEST_LEN)) {
- /* This one showed up in 3 digests. Twice with ID 'M', once with 'Z'. */
- tt_mem_op(rs->identity_digest,OP_EQ,
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5",
- DIGEST_LEN);
- tt_str_op(rs->nickname,OP_EQ, "router1");
- tt_mem_op(rs->descriptor_digest,OP_EQ, "MMMMMMMMMMMMMMMMMMMM", DIGEST_LEN);
- tt_int_op(rs->published_on,OP_EQ, now-1000);
- tt_int_op(rs->addr,OP_EQ, 0x99009901);
- tt_int_op(rs->or_port,OP_EQ, 443);
- tt_int_op(rs->dir_port,OP_EQ, 0);
- tor_addr_parse(&addr_ipv6, "[1:2:3::4]");
- tt_assert(tor_addr_eq(&rs->ipv6_addr, &addr_ipv6));
- tt_int_op(rs->ipv6_orport,OP_EQ, 4711);
- tt_assert(!rs->is_authority);
- tt_assert(rs->is_exit);
- tt_assert(rs->is_fast);
- tt_assert(rs->is_possible_guard);
- tt_assert(rs->is_stable);
- tt_assert(rs->is_flagged_running);
- tt_assert(rs->is_valid);
- tt_assert(!rs->is_named);
- /* XXXX check version */
- } else {
- /* Weren't expecting this... */
- tt_assert(0);
- }
- done:
- return;
- }
- /** Run a unit tests for generating and parsing networkstatuses, with
- * the supply test fns. */
- static void
- test_a_networkstatus(
- vote_routerstatus_t * (*vrs_gen)(int idx, time_t now),
- int (*vote_tweaks)(networkstatus_t *v, int voter, time_t now),
- void (*vrs_test)(vote_routerstatus_t *vrs, int voter, time_t now),
- void (*consensus_test)(networkstatus_t *con, time_t now),
- void (*rs_test)(routerstatus_t *rs, time_t now))
- {
- authority_cert_t *cert1=NULL, *cert2=NULL, *cert3=NULL;
- crypto_pk_t *sign_skey_1=NULL, *sign_skey_2=NULL, *sign_skey_3=NULL;
- crypto_pk_t *sign_skey_leg1=NULL;
- const char *msg=NULL;
- /*
- * Sum the non-zero returns from vote_tweaks() we've seen; if vote_tweaks()
- * returns non-zero, it changed net_params and we should skip the tests for
- * that later as they will fail.
- */
- int params_tweaked = 0;
- time_t now = time(NULL);
- networkstatus_voter_info_t *voter;
- document_signature_t *sig;
- networkstatus_t *vote=NULL, *v1=NULL, *v2=NULL, *v3=NULL, *con=NULL,
- *con_md=NULL;
- vote_routerstatus_t *vrs;
- routerstatus_t *rs;
- int idx, n_rs, n_vrs;
- char *v1_text=NULL, *v2_text=NULL, *v3_text=NULL, *consensus_text=NULL,
- *cp=NULL;
- smartlist_t *votes = smartlist_new();
- /* For generating the two other consensuses. */
- char *detached_text1=NULL, *detached_text2=NULL;
- char *consensus_text2=NULL, *consensus_text3=NULL;
- char *consensus_text_md2=NULL, *consensus_text_md3=NULL;
- char *consensus_text_md=NULL;
- networkstatus_t *con2=NULL, *con_md2=NULL, *con3=NULL, *con_md3=NULL;
- ns_detached_signatures_t *dsig1=NULL, *dsig2=NULL;
- tt_assert(vrs_gen);
- tt_assert(rs_test);
- tt_assert(vrs_test);
- /* Parse certificates and keys. */
- cert1 = authority_cert_parse_from_string(AUTHORITY_CERT_1, NULL);
- tt_assert(cert1);
- cert2 = authority_cert_parse_from_string(AUTHORITY_CERT_2, NULL);
- tt_assert(cert2);
- cert3 = authority_cert_parse_from_string(AUTHORITY_CERT_3, NULL);
- tt_assert(cert3);
- sign_skey_1 = crypto_pk_new();
- sign_skey_2 = crypto_pk_new();
- sign_skey_3 = crypto_pk_new();
- sign_skey_leg1 = pk_generate(4);
- tt_assert(!crypto_pk_read_private_key_from_string(sign_skey_1,
- AUTHORITY_SIGNKEY_1, -1));
- tt_assert(!crypto_pk_read_private_key_from_string(sign_skey_2,
- AUTHORITY_SIGNKEY_2, -1));
- tt_assert(!crypto_pk_read_private_key_from_string(sign_skey_3,
- AUTHORITY_SIGNKEY_3, -1));
- tt_assert(!crypto_pk_cmp_keys(sign_skey_1, cert1->signing_key));
- tt_assert(!crypto_pk_cmp_keys(sign_skey_2, cert2->signing_key));
- /*
- * Set up a vote; generate it; try to parse it.
- */
- vote = tor_malloc_zero(sizeof(networkstatus_t));
- vote->type = NS_TYPE_VOTE;
- vote->published = now;
- vote->valid_after = now+1000;
- vote->fresh_until = now+2000;
- vote->valid_until = now+3000;
- vote->vote_seconds = 100;
- vote->dist_seconds = 200;
- vote->supported_methods = smartlist_new();
- smartlist_split_string(vote->supported_methods, "1 2 3", NULL, 0, -1);
- vote->client_versions = tor_strdup("0.1.2.14,0.1.2.15");
- vote->server_versions = tor_strdup("0.1.2.14,0.1.2.15,0.1.2.16");
- vote->known_flags = smartlist_new();
- smartlist_split_string(vote->known_flags,
- "Authority Exit Fast Guard Running Stable V2Dir Valid",
- 0, SPLIT_SKIP_SPACE|SPLIT_IGNORE_BLANK, 0);
- vote->voters = smartlist_new();
- voter = tor_malloc_zero(sizeof(networkstatus_voter_info_t));
- voter->nickname = tor_strdup("Voter1");
- voter->address = tor_strdup("1.2.3.4");
- voter->addr = 0x01020304;
- voter->dir_port = 80;
- voter->or_port = 9000;
- voter->contact = tor_strdup("voter@example.com");
- crypto_pk_get_digest(cert1->identity_key, voter->identity_digest);
- smartlist_add(vote->voters, voter);
- vote->cert = authority_cert_dup(cert1);
- vote->net_params = smartlist_new();
- smartlist_split_string(vote->net_params, "circuitwindow=101 foo=990",
- NULL, 0, 0);
- vote->routerstatus_list = smartlist_new();
- /* add routerstatuses */
- idx = 0;
- do {
- vrs = vrs_gen(idx, now);
- if (vrs) {
- smartlist_add(vote->routerstatus_list, vrs);
- tt_assert(router_add_to_routerlist(generate_ri_from_rs(vrs),
- &msg,0,0)>=0);
- ++idx;
- }
- } while (vrs);
- n_vrs = idx;
- /* dump the vote and try to parse it. */
- v1_text = format_networkstatus_vote(sign_skey_1, vote);
- tt_assert(v1_text);
- v1 = networkstatus_parse_vote_from_string(v1_text, NULL, NS_TYPE_VOTE);
- tt_assert(v1);
- /* Make sure the parsed thing was right. */
- tt_int_op(v1->type,OP_EQ, NS_TYPE_VOTE);
- tt_int_op(v1->published,OP_EQ, vote->published);
- tt_int_op(v1->valid_after,OP_EQ, vote->valid_after);
- tt_int_op(v1->fresh_until,OP_EQ, vote->fresh_until);
- tt_int_op(v1->valid_until,OP_EQ, vote->valid_until);
- tt_int_op(v1->vote_seconds,OP_EQ, vote->vote_seconds);
- tt_int_op(v1->dist_seconds,OP_EQ, vote->dist_seconds);
- tt_str_op(v1->client_versions,OP_EQ, vote->client_versions);
- tt_str_op(v1->server_versions,OP_EQ, vote->server_versions);
- tt_assert(v1->voters && smartlist_len(v1->voters));
- voter = smartlist_get(v1->voters, 0);
- tt_str_op(voter->nickname,OP_EQ, "Voter1");
- tt_str_op(voter->address,OP_EQ, "1.2.3.4");
- tt_int_op(voter->addr,OP_EQ, 0x01020304);
- tt_int_op(voter->dir_port,OP_EQ, 80);
- tt_int_op(voter->or_port,OP_EQ, 9000);
- tt_str_op(voter->contact,OP_EQ, "voter@example.com");
- tt_assert(v1->cert);
- tt_assert(!crypto_pk_cmp_keys(sign_skey_1, v1->cert->signing_key));
- cp = smartlist_join_strings(v1->known_flags, ":", 0, NULL);
- tt_str_op(cp,OP_EQ, "Authority:Exit:Fast:Guard:Running:Stable:V2Dir:Valid");
- tor_free(cp);
- tt_int_op(smartlist_len(v1->routerstatus_list),OP_EQ, n_vrs);
- if (vote_tweaks) params_tweaked += vote_tweaks(v1, 1, now);
- /* Check the routerstatuses. */
- for (idx = 0; idx < n_vrs; ++idx) {
- vrs = smartlist_get(v1->routerstatus_list, idx);
- tt_assert(vrs);
- vrs_test(vrs, 1, now);
- }
- /* Generate second vote. It disagrees on some of the times,
- * and doesn't list versions, and knows some crazy flags */
- vote->published = now+1;
- vote->fresh_until = now+3005;
- vote->dist_seconds = 300;
- authority_cert_free(vote->cert);
- vote->cert = authority_cert_dup(cert2);
- SMARTLIST_FOREACH(vote->net_params, char *, c, tor_free(c));
- smartlist_clear(vote->net_params);
- smartlist_split_string(vote->net_params, "bar=2000000000 circuitwindow=20",
- NULL, 0, 0);
- tor_free(vote->client_versions);
- tor_free(vote->server_versions);
- voter = smartlist_get(vote->voters, 0);
- tor_free(voter->nickname);
- tor_free(voter->address);
- voter->nickname = tor_strdup("Voter2");
- voter->address = tor_strdup("2.3.4.5");
- voter->addr = 0x02030405;
- crypto_pk_get_digest(cert2->identity_key, voter->identity_digest);
- smartlist_add(vote->known_flags, tor_strdup("MadeOfCheese"));
- smartlist_add(vote->known_flags, tor_strdup("MadeOfTin"));
- smartlist_sort_strings(vote->known_flags);
- /* generate and parse v2. */
- v2_text = format_networkstatus_vote(sign_skey_2, vote);
- tt_assert(v2_text);
- v2 = networkstatus_parse_vote_from_string(v2_text, NULL, NS_TYPE_VOTE);
- tt_assert(v2);
- if (vote_tweaks) params_tweaked += vote_tweaks(v2, 2, now);
- /* Check that flags come out right.*/
- cp = smartlist_join_strings(v2->known_flags, ":", 0, NULL);
- tt_str_op(cp,OP_EQ, "Authority:Exit:Fast:Guard:MadeOfCheese:MadeOfTin:"
- "Running:Stable:V2Dir:Valid");
- tor_free(cp);
- /* Check the routerstatuses. */
- n_vrs = smartlist_len(v2->routerstatus_list);
- for (idx = 0; idx < n_vrs; ++idx) {
- vrs = smartlist_get(v2->routerstatus_list, idx);
- tt_assert(vrs);
- vrs_test(vrs, 2, now);
- }
- /* Generate the third vote. */
- vote->published = now;
- vote->fresh_until = now+2003;
- vote->dist_seconds = 250;
- authority_cert_free(vote->cert);
- vote->cert = authority_cert_dup(cert3);
- SMARTLIST_FOREACH(vote->net_params, char *, c, tor_free(c));
- smartlist_clear(vote->net_params);
- smartlist_split_string(vote->net_params, "circuitwindow=80 foo=660",
- NULL, 0, 0);
- smartlist_add(vote->supported_methods, tor_strdup("4"));
- vote->client_versions = tor_strdup("0.1.2.14,0.1.2.17");
- vote->server_versions = tor_strdup("0.1.2.10,0.1.2.15,0.1.2.16");
- voter = smartlist_get(vote->voters, 0);
- tor_free(voter->nickname);
- tor_free(voter->address);
- voter->nickname = tor_strdup("Voter3");
- voter->address = tor_strdup("3.4.5.6");
- voter->addr = 0x03040506;
- crypto_pk_get_digest(cert3->identity_key, voter->identity_digest);
- /* This one has a legacy id. */
- memset(voter->legacy_id_digest, (int)'A', DIGEST_LEN);
- v3_text = format_networkstatus_vote(sign_skey_3, vote);
- tt_assert(v3_text);
- v3 = networkstatus_parse_vote_from_string(v3_text, NULL, NS_TYPE_VOTE);
- tt_assert(v3);
- if (vote_tweaks) params_tweaked += vote_tweaks(v3, 3, now);
- /* Compute a consensus as voter 3. */
- smartlist_add(votes, v3);
- smartlist_add(votes, v1);
- smartlist_add(votes, v2);
- consensus_text = networkstatus_compute_consensus(votes, 3,
- cert3->identity_key,
- sign_skey_3,
- "AAAAAAAAAAAAAAAAAAAA",
- sign_skey_leg1,
- FLAV_NS);
- tt_assert(consensus_text);
- con = networkstatus_parse_vote_from_string(consensus_text, NULL,
- NS_TYPE_CONSENSUS);
- tt_assert(con);
- //log_notice(LD_GENERAL, "<<%s>>\n<<%s>>\n<<%s>>\n",
- // v1_text, v2_text, v3_text);
- consensus_text_md = networkstatus_compute_consensus(votes, 3,
- cert3->identity_key,
- sign_skey_3,
- "AAAAAAAAAAAAAAAAAAAA",
- sign_skey_leg1,
- FLAV_MICRODESC);
- tt_assert(consensus_text_md);
- con_md = networkstatus_parse_vote_from_string(consensus_text_md, NULL,
- NS_TYPE_CONSENSUS);
- tt_assert(con_md);
- tt_int_op(con_md->flavor,OP_EQ, FLAV_MICRODESC);
- /* Check consensus contents. */
- tt_assert(con->type == NS_TYPE_CONSENSUS);
- tt_int_op(con->published,OP_EQ, 0); /* this field only appears in votes. */
- tt_int_op(con->valid_after,OP_EQ, now+1000);
- tt_int_op(con->fresh_until,OP_EQ, now+2003); /* median */
- tt_int_op(con->valid_until,OP_EQ, now+3000);
- tt_int_op(con->vote_seconds,OP_EQ, 100);
- tt_int_op(con->dist_seconds,OP_EQ, 250); /* median */
- tt_str_op(con->client_versions,OP_EQ, "0.1.2.14");
- tt_str_op(con->server_versions,OP_EQ, "0.1.2.15,0.1.2.16");
- cp = smartlist_join_strings(v2->known_flags, ":", 0, NULL);
- tt_str_op(cp,OP_EQ, "Authority:Exit:Fast:Guard:MadeOfCheese:MadeOfTin:"
- "Running:Stable:V2Dir:Valid");
- tor_free(cp);
- if (!params_tweaked) {
- /* Skip this one if vote_tweaks() messed with the param lists */
- cp = smartlist_join_strings(con->net_params, ":", 0, NULL);
- tt_str_op(cp,OP_EQ, "circuitwindow=80:foo=660");
- tor_free(cp);
- }
- tt_int_op(4,OP_EQ, smartlist_len(con->voters)); /*3 voters, 1 legacy key.*/
- /* The voter id digests should be in this order. */
- tt_assert(memcmp(cert2->cache_info.identity_digest,
- cert1->cache_info.identity_digest,DIGEST_LEN)<0);
- tt_assert(memcmp(cert1->cache_info.identity_digest,
- cert3->cache_info.identity_digest,DIGEST_LEN)<0);
- test_same_voter(smartlist_get(con->voters, 1),
- smartlist_get(v2->voters, 0));
- test_same_voter(smartlist_get(con->voters, 2),
- smartlist_get(v1->voters, 0));
- test_same_voter(smartlist_get(con->voters, 3),
- smartlist_get(v3->voters, 0));
- consensus_test(con, now);
- /* Check the routerstatuses. */
- n_rs = smartlist_len(con->routerstatus_list);
- for (idx = 0; idx < n_rs; ++idx) {
- rs = smartlist_get(con->routerstatus_list, idx);
- tt_assert(rs);
- rs_test(rs, now);
- }
- /* Check signatures. the first voter is a pseudo-entry with a legacy key.
- * The second one hasn't signed. The fourth one has signed: validate it. */
- voter = smartlist_get(con->voters, 1);
- tt_int_op(smartlist_len(voter->sigs),OP_EQ, 0);
- voter = smartlist_get(con->voters, 3);
- tt_int_op(smartlist_len(voter->sigs),OP_EQ, 1);
- sig = smartlist_get(voter->sigs, 0);
- tt_assert(sig->signature);
- tt_assert(!sig->good_signature);
- tt_assert(!sig->bad_signature);
- tt_assert(!networkstatus_check_document_signature(con, sig, cert3));
- tt_assert(sig->signature);
- tt_assert(sig->good_signature);
- tt_assert(!sig->bad_signature);
- {
- const char *msg=NULL;
- /* Compute the other two signed consensuses. */
- smartlist_shuffle(votes);
- consensus_text2 = networkstatus_compute_consensus(votes, 3,
- cert2->identity_key,
- sign_skey_2, NULL,NULL,
- FLAV_NS);
- consensus_text_md2 = networkstatus_compute_consensus(votes, 3,
- cert2->identity_key,
- sign_skey_2, NULL,NULL,
- FLAV_MICRODESC);
- smartlist_shuffle(votes);
- consensus_text3 = networkstatus_compute_consensus(votes, 3,
- cert1->identity_key,
- sign_skey_1, NULL,NULL,
- FLAV_NS);
- consensus_text_md3 = networkstatus_compute_consensus(votes, 3,
- cert1->identity_key,
- sign_skey_1, NULL,NULL,
- FLAV_MICRODESC);
- tt_assert(consensus_text2);
- tt_assert(consensus_text3);
- tt_assert(consensus_text_md2);
- tt_assert(consensus_text_md3);
- con2 = networkstatus_parse_vote_from_string(consensus_text2, NULL,
- NS_TYPE_CONSENSUS);
- con3 = networkstatus_parse_vote_from_string(consensus_text3, NULL,
- NS_TYPE_CONSENSUS);
- con_md2 = networkstatus_parse_vote_from_string(consensus_text_md2, NULL,
- NS_TYPE_CONSENSUS);
- con_md3 = networkstatus_parse_vote_from_string(consensus_text_md3, NULL,
- NS_TYPE_CONSENSUS);
- tt_assert(con2);
- tt_assert(con3);
- tt_assert(con_md2);
- tt_assert(con_md3);
- /* All three should have the same digest. */
- tt_mem_op(&con->digests,OP_EQ, &con2->digests, sizeof(digests_t));
- tt_mem_op(&con->digests,OP_EQ, &con3->digests, sizeof(digests_t));
- tt_mem_op(&con_md->digests,OP_EQ, &con_md2->digests, sizeof(digests_t));
- tt_mem_op(&con_md->digests,OP_EQ, &con_md3->digests, sizeof(digests_t));
- /* Extract a detached signature from con3. */
- detached_text1 = get_detached_sigs(con3, con_md3);
- tt_assert(detached_text1);
- /* Try to parse it. */
- dsig1 = networkstatus_parse_detached_signatures(detached_text1, NULL);
- tt_assert(dsig1);
- /* Are parsed values as expected? */
- tt_int_op(dsig1->valid_after,OP_EQ, con3->valid_after);
- tt_int_op(dsig1->fresh_until,OP_EQ, con3->fresh_until);
- tt_int_op(dsig1->valid_until,OP_EQ, con3->valid_until);
- {
- digests_t *dsig_digests = strmap_get(dsig1->digests, "ns");
- tt_assert(dsig_digests);
- tt_mem_op(dsig_digests->d[DIGEST_SHA1], OP_EQ,
- con3->digests.d[DIGEST_SHA1], DIGEST_LEN);
- dsig_digests = strmap_get(dsig1->digests, "microdesc");
- tt_assert(dsig_digests);
- tt_mem_op(dsig_digests->d[DIGEST_SHA256],OP_EQ,
- con_md3->digests.d[DIGEST_SHA256],
- DIGEST256_LEN);
- }
- {
- smartlist_t *dsig_signatures = strmap_get(dsig1->signatures, "ns");
- tt_assert(dsig_signatures);
- tt_int_op(1,OP_EQ, smartlist_len(dsig_signatures));
- sig = smartlist_get(dsig_signatures, 0);
- tt_mem_op(sig->identity_digest,OP_EQ, cert1->cache_info.identity_digest,
- DIGEST_LEN);
- tt_int_op(sig->alg,OP_EQ, DIGEST_SHA1);
- dsig_signatures = strmap_get(dsig1->signatures, "microdesc");
- tt_assert(dsig_signatures);
- tt_int_op(1,OP_EQ, smartlist_len(dsig_signatures));
- sig = smartlist_get(dsig_signatures, 0);
- tt_mem_op(sig->identity_digest,OP_EQ, cert1->cache_info.identity_digest,
- DIGEST_LEN);
- tt_int_op(sig->alg,OP_EQ, DIGEST_SHA256);
- }
- /* Try adding it to con2. */
- detached_text2 = get_detached_sigs(con2,con_md2);
- tt_int_op(1,OP_EQ, networkstatus_add_detached_signatures(con2, dsig1,
- "test", LOG_INFO, &msg));
- tor_free(detached_text2);
- tt_int_op(1,OP_EQ,
- networkstatus_add_detached_signatures(con_md2, dsig1, "test",
- LOG_INFO, &msg));
- tor_free(detached_text2);
- detached_text2 = get_detached_sigs(con2,con_md2);
- //printf("\n<%s>\n", detached_text2);
- dsig2 = networkstatus_parse_detached_signatures(detached_text2, NULL);
- tt_assert(dsig2);
- /*
- printf("\n");
- SMARTLIST_FOREACH(dsig2->signatures, networkstatus_voter_info_t *, vi, {
- char hd[64];
- base16_encode(hd, sizeof(hd), vi->identity_digest, DIGEST_LEN);
- printf("%s\n", hd);
- });
- */
- tt_int_op(2,OP_EQ,
- smartlist_len((smartlist_t*)strmap_get(dsig2->signatures, "ns")));
- tt_int_op(2,OP_EQ,
- smartlist_len((smartlist_t*)strmap_get(dsig2->signatures,
- "microdesc")));
- /* Try adding to con2 twice; verify that nothing changes. */
- tt_int_op(0,OP_EQ, networkstatus_add_detached_signatures(con2, dsig1,
- "test", LOG_INFO, &msg));
- /* Add to con. */
- tt_int_op(2,OP_EQ, networkstatus_add_detached_signatures(con, dsig2,
- "test", LOG_INFO, &msg));
- /* Check signatures */
- voter = smartlist_get(con->voters, 1);
- sig = smartlist_get(voter->sigs, 0);
- tt_assert(sig);
- tt_assert(!networkstatus_check_document_signature(con, sig, cert2));
- voter = smartlist_get(con->voters, 2);
- sig = smartlist_get(voter->sigs, 0);
- tt_assert(sig);
- tt_assert(!networkstatus_check_document_signature(con, sig, cert1));
- }
- done:
- tor_free(cp);
- smartlist_free(votes);
- tor_free(v1_text);
- tor_free(v2_text);
- tor_free(v3_text);
- tor_free(consensus_text);
- tor_free(consensus_text_md);
- if (vote)
- networkstatus_vote_free(vote);
- if (v1)
- networkstatus_vote_free(v1);
- if (v2)
- networkstatus_vote_free(v2);
- if (v3)
- networkstatus_vote_free(v3);
- if (con)
- networkstatus_vote_free(con);
- if (con_md)
- networkstatus_vote_free(con_md);
- if (sign_skey_1)
- crypto_pk_free(sign_skey_1);
- if (sign_skey_2)
- crypto_pk_free(sign_skey_2);
- if (sign_skey_3)
- crypto_pk_free(sign_skey_3);
- if (sign_skey_leg1)
- crypto_pk_free(sign_skey_leg1);
- if (cert1)
- authority_cert_free(cert1);
- if (cert2)
- authority_cert_free(cert2);
- if (cert3)
- authority_cert_free(cert3);
- tor_free(consensus_text2);
- tor_free(consensus_text3);
- tor_free(consensus_text_md2);
- tor_free(consensus_text_md3);
- tor_free(detached_text1);
- tor_free(detached_text2);
- if (con2)
- networkstatus_vote_free(con2);
- if (con3)
- networkstatus_vote_free(con3);
- if (con_md2)
- networkstatus_vote_free(con_md2);
- if (con_md3)
- networkstatus_vote_free(con_md3);
- if (dsig1)
- ns_detached_signatures_free(dsig1);
- if (dsig2)
- ns_detached_signatures_free(dsig2);
- }
- /** Run unit tests for generating and parsing V3 consensus networkstatus
- * documents. */
- static void
- test_dir_v3_networkstatus(void *arg)
- {
- (void)arg;
- test_a_networkstatus(gen_routerstatus_for_v3ns,
- vote_tweaks_for_v3ns,
- test_vrs_for_v3ns,
- test_consensus_for_v3ns,
- test_routerstatus_for_v3ns);
- }
- static void
- test_dir_scale_bw(void *testdata)
- {
- double v[8] = { 2.0/3,
- 7.0,
- 1.0,
- 3.0,
- 1.0/5,
- 1.0/7,
- 12.0,
- 24.0 };
- u64_dbl_t vals[8];
- uint64_t total;
- int i;
- (void) testdata;
- for (i=0; i<8; ++i)
- vals[i].dbl = v[i];
- scale_array_elements_to_u64(vals, 8, &total);
- tt_int_op((int)total, OP_EQ, 48);
- total = 0;
- for (i=0; i<8; ++i) {
- total += vals[i].u64;
- }
- tt_assert(total >= (U64_LITERAL(1)<<60));
- tt_assert(total <= (U64_LITERAL(1)<<62));
- for (i=0; i<8; ++i) {
- /* vals[2].u64 is the scaled value of 1.0 */
- double ratio = ((double)vals[i].u64) / vals[2].u64;
- tt_double_op(fabs(ratio - v[i]), OP_LT, .00001);
- }
- /* test handling of no entries */
- total = 1;
- scale_array_elements_to_u64(vals, 0, &total);
- tt_assert(total == 0);
- /* make sure we don't read the array when we have no entries
- * may require compiler flags to catch NULL dereferences */
- total = 1;
- scale_array_elements_to_u64(NULL, 0, &total);
- tt_assert(total == 0);
- scale_array_elements_to_u64(NULL, 0, NULL);
- /* test handling of zero totals */
- total = 1;
- vals[0].dbl = 0.0;
- scale_array_elements_to_u64(vals, 1, &total);
- tt_assert(total == 0);
- tt_assert(vals[0].u64 == 0);
- vals[0].dbl = 0.0;
- vals[1].dbl = 0.0;
- scale_array_elements_to_u64(vals, 2, NULL);
- tt_assert(vals[0].u64 == 0);
- tt_assert(vals[1].u64 == 0);
- done:
- ;
- }
- static void
- test_dir_random_weighted(void *testdata)
- {
- int histogram[10];
- uint64_t vals[10] = {3,1,2,4,6,0,7,5,8,9}, total=0;
- u64_dbl_t inp[10];
- int i, choice;
- const int n = 50000;
- double max_sq_error;
- (void) testdata;
- /* Try a ten-element array with values from 0 through 10. The values are
- * in a scrambled order to make sure we don't depend on order. */
- memset(histogram,0,sizeof(histogram));
- for (i=0; i<10; ++i) {
- inp[i].u64 = vals[i];
- total += vals[i];
- }
- tt_u64_op(total, OP_EQ, 45);
- for (i=0; i<n; ++i) {
- choice = choose_array_element_by_weight(inp, 10);
- tt_int_op(choice, OP_GE, 0);
- tt_int_op(choice, OP_LT, 10);
- histogram[choice]++;
- }
- /* Now see if we chose things about frequently enough. */
- max_sq_error = 0;
- for (i=0; i<10; ++i) {
- int expected = (int)(n*vals[i]/total);
- double frac_diff = 0, sq;
- TT_BLATHER((" %d : %5d vs %5d\n", (int)vals[i], histogram[i], expected));
- if (expected)
- frac_diff = (histogram[i] - expected) / ((double)expected);
- else
- tt_int_op(histogram[i], OP_EQ, 0);
- sq = frac_diff * frac_diff;
- if (sq > max_sq_error)
- max_sq_error = sq;
- }
- /* It should almost always be much much less than this. If you want to
- * figure out the odds, please feel free. */
- tt_double_op(max_sq_error, OP_LT, .05);
- /* Now try a singleton; do we choose it? */
- for (i = 0; i < 100; ++i) {
- choice = choose_array_element_by_weight(inp, 1);
- tt_int_op(choice, OP_EQ, 0);
- }
- /* Now try an array of zeros. We should choose randomly. */
- memset(histogram,0,sizeof(histogram));
- for (i = 0; i < 5; ++i)
- inp[i].u64 = 0;
- for (i = 0; i < n; ++i) {
- choice = choose_array_element_by_weight(inp, 5);
- tt_int_op(choice, OP_GE, 0);
- tt_int_op(choice, OP_LT, 5);
- histogram[choice]++;
- }
- /* Now see if we chose things about frequently enough. */
- max_sq_error = 0;
- for (i=0; i<5; ++i) {
- int expected = n/5;
- double frac_diff = 0, sq;
- TT_BLATHER((" %d : %5d vs %5d\n", (int)vals[i], histogram[i], expected));
- frac_diff = (histogram[i] - expected) / ((double)expected);
- sq = frac_diff * frac_diff;
- if (sq > max_sq_error)
- max_sq_error = sq;
- }
- /* It should almost always be much much less than this. If you want to
- * figure out the odds, please feel free. */
- tt_double_op(max_sq_error, OP_LT, .05);
- done:
- ;
- }
- /* Function pointers for test_dir_clip_unmeasured_bw_kb() */
- static uint32_t alternate_clip_bw = 0;
- /**
- * Generate a routerstatus for clip_unmeasured_bw_kb test; based on the
- * v3_networkstatus ones.
- */
- static vote_routerstatus_t *
- gen_routerstatus_for_umbw(int idx, time_t now)
- {
- vote_routerstatus_t *vrs = NULL;
- routerstatus_t *rs;
- tor_addr_t addr_ipv6;
- uint32_t max_unmeasured_bw_kb = (alternate_clip_bw > 0) ?
- alternate_clip_bw : DEFAULT_MAX_UNMEASURED_BW_KB;
- switch (idx) {
- case 0:
- /* Generate the first routerstatus. */
- vrs = tor_malloc_zero(sizeof(vote_routerstatus_t));
- rs = &vrs->status;
- vrs->version = tor_strdup("0.1.2.14");
- rs->published_on = now-1500;
- strlcpy(rs->nickname, "router2", sizeof(rs->nickname));
- memset(rs->identity_digest, 3, DIGEST_LEN);
- memset(rs->descriptor_digest, 78, DIGEST_LEN);
- rs->addr = 0x99008801;
- rs->or_port = 443;
- rs->dir_port = 8000;
- /* all flags but running cleared */
- rs->is_flagged_running = 1;
- /*
- * This one has measured bandwidth below the clip cutoff, and
- * so shouldn't be clipped; we'll have to test that it isn't
- * later.
- */
- vrs->has_measured_bw = 1;
- rs->has_bandwidth = 1;
- vrs->measured_bw_kb = rs->bandwidth_kb = max_unmeasured_bw_kb / 2;
- break;
- case 1:
- /* Generate the second routerstatus. */
- vrs = tor_malloc_zero(sizeof(vote_routerstatus_t));
- rs = &vrs->status;
- vrs->version = tor_strdup("0.2.0.5");
- rs->published_on = now-1000;
- strlcpy(rs->nickname, "router1", sizeof(rs->nickname));
- memset(rs->identity_digest, 5, DIGEST_LEN);
- memset(rs->descriptor_digest, 77, DIGEST_LEN);
- rs->addr = 0x99009901;
- rs->or_port = 443;
- rs->dir_port = 0;
- tor_addr_parse(&addr_ipv6, "[1:2:3::4]");
- tor_addr_copy(&rs->ipv6_addr, &addr_ipv6);
- rs->ipv6_orport = 4711;
- rs->is_exit = rs->is_stable = rs->is_fast = rs->is_flagged_running =
- rs->is_valid = rs->is_possible_guard = 1;
- /*
- * This one has measured bandwidth above the clip cutoff, and
- * so shouldn't be clipped; we'll have to test that it isn't
- * later.
- */
- vrs->has_measured_bw = 1;
- rs->has_bandwidth = 1;
- vrs->measured_bw_kb = rs->bandwidth_kb = 2 * max_unmeasured_bw_kb;
- break;
- case 2:
- /* Generate the third routerstatus. */
- vrs = tor_malloc_zero(sizeof(vote_routerstatus_t));
- rs = &vrs->status;
- vrs->version = tor_strdup("0.1.0.3");
- rs->published_on = now-1000;
- strlcpy(rs->nickname, "router3", sizeof(rs->nickname));
- memset(rs->identity_digest, 0x33, DIGEST_LEN);
- memset(rs->descriptor_digest, 79, DIGEST_LEN);
- rs->addr = 0xAA009901;
- rs->or_port = 400;
- rs->dir_port = 9999;
- rs->is_authority = rs->is_exit = rs->is_stable = rs->is_fast =
- rs->is_flagged_running = rs->is_valid =
- rs->is_possible_guard = 1;
- /*
- * This one has unmeasured bandwidth above the clip cutoff, and
- * so should be clipped; we'll have to test that it isn't
- * later.
- */
- vrs->has_measured_bw = 0;
- rs->has_bandwidth = 1;
- vrs->measured_bw_kb = 0;
- rs->bandwidth_kb = 2 * max_unmeasured_bw_kb;
- break;
- case 3:
- /* Generate a fourth routerstatus that is not running. */
- vrs = tor_malloc_zero(sizeof(vote_routerstatus_t));
- rs = &vrs->status;
- vrs->version = tor_strdup("0.1.6.3");
- rs->published_on = now-1000;
- strlcpy(rs->nickname, "router4", sizeof(rs->nickname));
- memset(rs->identity_digest, 0x34, DIGEST_LEN);
- memset(rs->descriptor_digest, 47, DIGEST_LEN);
- rs->addr = 0xC0000203;
- rs->or_port = 500;
- rs->dir_port = 1999;
- /* all flags but running cleared */
- rs->is_flagged_running = 1;
- /*
- * This one has unmeasured bandwidth below the clip cutoff, and
- * so shouldn't be clipped; we'll have to test that it isn't
- * later.
- */
- vrs->has_measured_bw = 0;
- rs->has_bandwidth = 1;
- vrs->measured_bw_kb = 0;
- rs->bandwidth_kb = max_unmeasured_bw_kb / 2;
- break;
- case 4:
- /* No more for this test; return NULL */
- vrs = NULL;
- break;
- default:
- /* Shouldn't happen */
- tt_assert(0);
- }
- if (vrs) {
- vrs->microdesc = tor_malloc_zero(sizeof(vote_microdesc_hash_t));
- tor_asprintf(&vrs->microdesc->microdesc_hash_line,
- "m 9,10,11,12,13,14,15,16,17 "
- "sha256=xyzajkldsdsajdadlsdjaslsdksdjlsdjsdaskdaaa%d\n",
- idx);
- }
- done:
- return vrs;
- }
- /** Apply tweaks to the vote list for each voter; for the umbw test this is
- * just adding the right consensus methods to let clipping happen */
- static int
- vote_tweaks_for_umbw(networkstatus_t *v, int voter, time_t now)
- {
- char *maxbw_param = NULL;
- int rv = 0;
- tt_assert(v);
- (void)voter;
- (void)now;
- tt_assert(v->supported_methods);
- SMARTLIST_FOREACH(v->supported_methods, char *, c, tor_free(c));
- smartlist_clear(v->supported_methods);
- /* Method 17 is MIN_METHOD_TO_CLIP_UNMEASURED_BW_KB */
- smartlist_split_string(v->supported_methods,
- "1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17",
- NULL, 0, -1);
- /* If we're using a non-default clip bandwidth, add it to net_params */
- if (alternate_clip_bw > 0) {
- tor_asprintf(&maxbw_param, "maxunmeasuredbw=%u", alternate_clip_bw);
- tt_assert(maxbw_param);
- if (maxbw_param) {
- smartlist_add(v->net_params, maxbw_param);
- rv = 1;
- }
- }
- done:
- return rv;
- }
- /**
- * Test a parsed vote_routerstatus_t for umbw test.
- */
- static void
- test_vrs_for_umbw(vote_routerstatus_t *vrs, int voter, time_t now)
- {
- routerstatus_t *rs;
- tor_addr_t addr_ipv6;
- uint32_t max_unmeasured_bw_kb = (alternate_clip_bw > 0) ?
- alternate_clip_bw : DEFAULT_MAX_UNMEASURED_BW_KB;
- (void)voter;
- tt_assert(vrs);
- rs = &(vrs->status);
- tt_assert(rs);
- /* Split out by digests to test */
- if (tor_memeq(rs->identity_digest,
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3"
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3",
- DIGEST_LEN)) {
- /*
- * Check the first routerstatus - measured bandwidth below the clip
- * cutoff.
- */
- tt_str_op(vrs->version,OP_EQ, "0.1.2.14");
- tt_int_op(rs->published_on,OP_EQ, now-1500);
- tt_str_op(rs->nickname,OP_EQ, "router2");
- tt_mem_op(rs->identity_digest,OP_EQ,
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3"
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3",
- DIGEST_LEN);
- tt_mem_op(rs->descriptor_digest,OP_EQ, "NNNNNNNNNNNNNNNNNNNN", DIGEST_LEN);
- tt_int_op(rs->addr,OP_EQ, 0x99008801);
- tt_int_op(rs->or_port,OP_EQ, 443);
- tt_int_op(rs->dir_port,OP_EQ, 8000);
- tt_assert(rs->has_bandwidth);
- tt_assert(vrs->has_measured_bw);
- tt_int_op(rs->bandwidth_kb,OP_EQ, max_unmeasured_bw_kb / 2);
- tt_int_op(vrs->measured_bw_kb,OP_EQ, max_unmeasured_bw_kb / 2);
- } else if (tor_memeq(rs->identity_digest,
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5"
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5",
- DIGEST_LEN)) {
- /*
- * Check the second routerstatus - measured bandwidth above the clip
- * cutoff.
- */
- tt_str_op(vrs->version,OP_EQ, "0.2.0.5");
- tt_int_op(rs->published_on,OP_EQ, now-1000);
- tt_str_op(rs->nickname,OP_EQ, "router1");
- tt_mem_op(rs->identity_digest,OP_EQ,
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5"
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5",
- DIGEST_LEN);
- tt_mem_op(rs->descriptor_digest,OP_EQ, "MMMMMMMMMMMMMMMMMMMM", DIGEST_LEN);
- tt_int_op(rs->addr,OP_EQ, 0x99009901);
- tt_int_op(rs->or_port,OP_EQ, 443);
- tt_int_op(rs->dir_port,OP_EQ, 0);
- tor_addr_parse(&addr_ipv6, "[1:2:3::4]");
- tt_assert(tor_addr_eq(&rs->ipv6_addr, &addr_ipv6));
- tt_int_op(rs->ipv6_orport,OP_EQ, 4711);
- tt_assert(rs->has_bandwidth);
- tt_assert(vrs->has_measured_bw);
- tt_int_op(rs->bandwidth_kb,OP_EQ, max_unmeasured_bw_kb * 2);
- tt_int_op(vrs->measured_bw_kb,OP_EQ, max_unmeasured_bw_kb * 2);
- } else if (tor_memeq(rs->identity_digest,
- "\x33\x33\x33\x33\x33\x33\x33\x33\x33\x33"
- "\x33\x33\x33\x33\x33\x33\x33\x33\x33\x33",
- DIGEST_LEN)) {
- /*
- * Check the third routerstatus - unmeasured bandwidth above the clip
- * cutoff; this one should be clipped later on in the consensus, but
- * appears unclipped in the vote.
- */
- tt_assert(rs->has_bandwidth);
- tt_assert(!(vrs->has_measured_bw));
- tt_int_op(rs->bandwidth_kb,OP_EQ, max_unmeasured_bw_kb * 2);
- tt_int_op(vrs->measured_bw_kb,OP_EQ, 0);
- } else if (tor_memeq(rs->identity_digest,
- "\x34\x34\x34\x34\x34\x34\x34\x34\x34\x34"
- "\x34\x34\x34\x34\x34\x34\x34\x34\x34\x34",
- DIGEST_LEN)) {
- /*
- * Check the fourth routerstatus - unmeasured bandwidth below the clip
- * cutoff; this one should not be clipped.
- */
- tt_assert(rs->has_bandwidth);
- tt_assert(!(vrs->has_measured_bw));
- tt_int_op(rs->bandwidth_kb,OP_EQ, max_unmeasured_bw_kb / 2);
- tt_int_op(vrs->measured_bw_kb,OP_EQ, 0);
- } else {
- tt_assert(0);
- }
- done:
- return;
- }
- /**
- * Test a consensus for v3_networkstatus_test
- */
- static void
- test_consensus_for_umbw(networkstatus_t *con, time_t now)
- {
- (void)now;
- tt_assert(con);
- tt_assert(!con->cert);
- // tt_assert(con->consensus_method >= MIN_METHOD_TO_CLIP_UNMEASURED_BW_KB);
- tt_assert(con->consensus_method >= 16);
- tt_int_op(4,OP_EQ, smartlist_len(con->routerstatus_list));
- /* There should be four listed routers; all voters saw the same in this */
- done:
- return;
- }
- /**
- * Test a router list entry for umbw test
- */
- static void
- test_routerstatus_for_umbw(routerstatus_t *rs, time_t now)
- {
- tor_addr_t addr_ipv6;
- uint32_t max_unmeasured_bw_kb = (alternate_clip_bw > 0) ?
- alternate_clip_bw : DEFAULT_MAX_UNMEASURED_BW_KB;
- tt_assert(rs);
- /* There should be four listed routers, as constructed above */
- if (tor_memeq(rs->identity_digest,
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3"
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3",
- DIGEST_LEN)) {
- tt_mem_op(rs->identity_digest,OP_EQ,
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3"
- "\x3\x3\x3\x3\x3\x3\x3\x3\x3\x3",
- DIGEST_LEN);
- tt_mem_op(rs->descriptor_digest,OP_EQ, "NNNNNNNNNNNNNNNNNNNN", DIGEST_LEN);
- tt_assert(!rs->is_authority);
- tt_assert(!rs->is_exit);
- tt_assert(!rs->is_fast);
- tt_assert(!rs->is_possible_guard);
- tt_assert(!rs->is_stable);
- /* (If it wasn't running it wouldn't be here) */
- tt_assert(rs->is_flagged_running);
- tt_assert(!rs->is_valid);
- tt_assert(!rs->is_named);
- /* This one should have measured bandwidth below the clip cutoff */
- tt_assert(rs->has_bandwidth);
- tt_int_op(rs->bandwidth_kb,OP_EQ, max_unmeasured_bw_kb / 2);
- tt_assert(!(rs->bw_is_unmeasured));
- } else if (tor_memeq(rs->identity_digest,
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5"
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5",
- DIGEST_LEN)) {
- /* This one showed up in 3 digests. Twice with ID 'M', once with 'Z'. */
- tt_mem_op(rs->identity_digest,OP_EQ,
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5"
- "\x5\x5\x5\x5\x5\x5\x5\x5\x5\x5",
- DIGEST_LEN);
- tt_str_op(rs->nickname,OP_EQ, "router1");
- tt_mem_op(rs->descriptor_digest,OP_EQ, "MMMMMMMMMMMMMMMMMMMM", DIGEST_LEN);
- tt_int_op(rs->published_on,OP_EQ, now-1000);
- tt_int_op(rs->addr,OP_EQ, 0x99009901);
- tt_int_op(rs->or_port,OP_EQ, 443);
- tt_int_op(rs->dir_port,OP_EQ, 0);
- tor_addr_parse(&addr_ipv6, "[1:2:3::4]");
- tt_assert(tor_addr_eq(&rs->ipv6_addr, &addr_ipv6));
- tt_int_op(rs->ipv6_orport,OP_EQ, 4711);
- tt_assert(!rs->is_authority);
- tt_assert(rs->is_exit);
- tt_assert(rs->is_fast);
- tt_assert(rs->is_possible_guard);
- tt_assert(rs->is_stable);
- tt_assert(rs->is_flagged_running);
- tt_assert(rs->is_valid);
- tt_assert(!rs->is_named);
- /* This one should have measured bandwidth above the clip cutoff */
- tt_assert(rs->has_bandwidth);
- tt_int_op(rs->bandwidth_kb,OP_EQ, max_unmeasured_bw_kb * 2);
- tt_assert(!(rs->bw_is_unmeasured));
- } else if (tor_memeq(rs->identity_digest,
- "\x33\x33\x33\x33\x33\x33\x33\x33\x33\x33"
- "\x33\x33\x33\x33\x33\x33\x33\x33\x33\x33",
- DIGEST_LEN)) {
- /*
- * This one should have unmeasured bandwidth above the clip cutoff,
- * and so should be clipped
- */
- tt_assert(rs->has_bandwidth);
- tt_int_op(rs->bandwidth_kb,OP_EQ, max_unmeasured_bw_kb);
- tt_assert(rs->bw_is_unmeasured);
- } else if (tor_memeq(rs->identity_digest,
- "\x34\x34\x34\x34\x34\x34\x34\x34\x34\x34"
- "\x34\x34\x34\x34\x34\x34\x34\x34\x34\x34",
- DIGEST_LEN)) {
- /*
- * This one should have unmeasured bandwidth below the clip cutoff,
- * and so should not be clipped
- */
- tt_assert(rs->has_bandwidth);
- tt_int_op(rs->bandwidth_kb,OP_EQ, max_unmeasured_bw_kb / 2);
- tt_assert(rs->bw_is_unmeasured);
- } else {
- /* Weren't expecting this... */
- tt_assert(0);
- }
- done:
- return;
- }
- /**
- * Compute a consensus involving clipping unmeasured bandwidth with consensus
- * method 17; this uses the same test_a_networkstatus() function that the
- * v3_networkstatus test uses.
- */
- static void
- test_dir_clip_unmeasured_bw_kb(void *arg)
- {
- /* Run the test with the default clip bandwidth */
- (void)arg;
- alternate_clip_bw = 0;
- test_a_networkstatus(gen_routerstatus_for_umbw,
- vote_tweaks_for_umbw,
- test_vrs_for_umbw,
- test_consensus_for_umbw,
- test_routerstatus_for_umbw);
- }
- /**
- * This version of test_dir_clip_unmeasured_bw_kb() uses a non-default choice
- * of clip bandwidth.
- */
- static void
- test_dir_clip_unmeasured_bw_kb_alt(void *arg)
- {
- /*
- * Try a different one; this value is chosen so that the below-the-cutoff
- * unmeasured nodes the test uses, at alternate_clip_bw / 2, will be above
- * DEFAULT_MAX_UNMEASURED_BW_KB and if the consensus incorrectly uses that
- * cutoff it will fail the test.
- */
- (void)arg;
- alternate_clip_bw = 3 * DEFAULT_MAX_UNMEASURED_BW_KB;
- test_a_networkstatus(gen_routerstatus_for_umbw,
- vote_tweaks_for_umbw,
- test_vrs_for_umbw,
- test_consensus_for_umbw,
- test_routerstatus_for_umbw);
- }
- static void
- test_dir_fmt_control_ns(void *arg)
- {
- char *s = NULL;
- routerstatus_t rs;
- (void)arg;
- memset(&rs, 0, sizeof(rs));
- rs.published_on = 1364925198;
- strlcpy(rs.nickname, "TetsuoMilk", sizeof(rs.nickname));
- memcpy(rs.identity_digest, "Stately, plump Buck ", DIGEST_LEN);
- memcpy(rs.descriptor_digest, "Mulligan came up fro", DIGEST_LEN);
- rs.addr = 0x20304050;
- rs.or_port = 9001;
- rs.dir_port = 9002;
- rs.is_exit = 1;
- rs.is_fast = 1;
- rs.is_flagged_running = 1;
- rs.has_bandwidth = 1;
- rs.bandwidth_kb = 1000;
- s = networkstatus_getinfo_helper_single(&rs);
- tt_assert(s);
- tt_str_op(s, OP_EQ,
- "r TetsuoMilk U3RhdGVseSwgcGx1bXAgQnVjayA "
- "TXVsbGlnYW4gY2FtZSB1cCBmcm8 2013-04-02 17:53:18 "
- "32.48.64.80 9001 9002\n"
- "s Exit Fast Running V2Dir\n"
- "w Bandwidth=1000\n");
- done:
- tor_free(s);
- }
- static void
- test_dir_http_handling(void *args)
- {
- char *url = NULL;
- (void)args;
- /* Parse http url tests: */
- /* Good headers */
- tt_int_op(parse_http_url("GET /tor/a/b/c.txt HTTP/1.1\r\n"
- "Host: example.com\r\n"
- "User-Agent: Mozilla/5.0 (Windows;"
- " U; Windows NT 6.1; en-US; rv:1.9.1.5)\r\n",
- &url),OP_EQ, 0);
- tt_str_op(url,OP_EQ, "/tor/a/b/c.txt");
- tor_free(url);
- tt_int_op(parse_http_url("GET /tor/a/b/c.txt HTTP/1.0\r\n", &url),OP_EQ, 0);
- tt_str_op(url,OP_EQ, "/tor/a/b/c.txt");
- tor_free(url);
- tt_int_op(parse_http_url("GET /tor/a/b/c.txt HTTP/1.600\r\n", &url),
- OP_EQ, 0);
- tt_str_op(url,OP_EQ, "/tor/a/b/c.txt");
- tor_free(url);
- /* Should prepend '/tor/' to url if required */
- tt_int_op(parse_http_url("GET /a/b/c.txt HTTP/1.1\r\n"
- "Host: example.com\r\n"
- "User-Agent: Mozilla/5.0 (Windows;"
- " U; Windows NT 6.1; en-US; rv:1.9.1.5)\r\n",
- &url),OP_EQ, 0);
- tt_str_op(url,OP_EQ, "/tor/a/b/c.txt");
- tor_free(url);
- /* Bad headers -- no HTTP/1.x*/
- tt_int_op(parse_http_url("GET /a/b/c.txt\r\n"
- "Host: example.com\r\n"
- "User-Agent: Mozilla/5.0 (Windows;"
- " U; Windows NT 6.1; en-US; rv:1.9.1.5)\r\n",
- &url),OP_EQ, -1);
- tt_assert(!url);
- /* Bad headers */
- tt_int_op(parse_http_url("GET /a/b/c.txt\r\n"
- "Host: example.com\r\n"
- "User-Agent: Mozilla/5.0 (Windows;"
- " U; Windows NT 6.1; en-US; rv:1.9.1.5)\r\n",
- &url),OP_EQ, -1);
- tt_assert(!url);
- tt_int_op(parse_http_url("GET /tor/a/b/c.txt", &url),OP_EQ, -1);
- tt_assert(!url);
- tt_int_op(parse_http_url("GET /tor/a/b/c.txt HTTP/1.1", &url),OP_EQ, -1);
- tt_assert(!url);
- tt_int_op(parse_http_url("GET /tor/a/b/c.txt HTTP/1.1x\r\n", &url),
- OP_EQ, -1);
- tt_assert(!url);
- tt_int_op(parse_http_url("GET /tor/a/b/c.txt HTTP/1.", &url),OP_EQ, -1);
- tt_assert(!url);
- tt_int_op(parse_http_url("GET /tor/a/b/c.txt HTTP/1.\r", &url),OP_EQ, -1);
- tt_assert(!url);
- done:
- tor_free(url);
- }
- #define DIR_LEGACY(name) \
- { #name, test_dir_ ## name , TT_FORK, NULL, NULL }
- #define DIR(name,flags) \
- { #name, test_dir_##name, (flags), NULL, NULL }
- struct testcase_t dir_tests[] = {
- DIR_LEGACY(nicknames),
- DIR_LEGACY(formats),
- DIR(routerparse_bad, 0),
- DIR(extrainfo_parsing, 0),
- DIR(parse_router_list, TT_FORK),
- DIR(load_routers, TT_FORK),
- DIR(load_extrainfo, TT_FORK),
- DIR_LEGACY(versions),
- DIR_LEGACY(fp_pairs),
- DIR(split_fps, 0),
- DIR_LEGACY(measured_bw_kb),
- DIR_LEGACY(measured_bw_kb_cache),
- DIR_LEGACY(param_voting),
- DIR_LEGACY(v3_networkstatus),
- DIR(random_weighted, 0),
- DIR(scale_bw, 0),
- DIR_LEGACY(clip_unmeasured_bw_kb),
- DIR_LEGACY(clip_unmeasured_bw_kb_alt),
- DIR(fmt_control_ns, 0),
- DIR(http_handling, 0),
- END_OF_TESTCASES
- };
|