123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896 |
- /* Copyright (c) 2017, The Tor Project, Inc. */
- /* See LICENSE for licensing information */
- #define CONSDIFFMGR_PRIVATE
- #include "or.h"
- #include "config.h"
- #include "conscache.h"
- #include "consdiff.h"
- #include "consdiffmgr.h"
- #include "cpuworker.h"
- #include "networkstatus.h"
- #include "routerparse.h"
- #include "workqueue.h"
- #include "test.h"
- #include "log_test_helpers.h"
- // ============================== Setup/teardown the consdiffmgr
- // These functions get run before/after each test in this module
- static void *
- consdiffmgr_test_setup(const struct testcase_t *arg)
- {
- (void)arg;
- char *ddir_fname = tor_strdup(get_fname_rnd("datadir_cdm"));
- tor_free(get_options_mutable()->DataDirectory);
- get_options_mutable()->DataDirectory = ddir_fname; // now owns the pointer.
- check_private_dir(ddir_fname, CPD_CREATE, NULL);
- consdiff_cfg_t consdiff_cfg = { 300 };
- consdiffmgr_configure(&consdiff_cfg);
- return (void *)1; // must return something non-null.
- }
- static int
- consdiffmgr_test_teardown(const struct testcase_t *arg, void *ignore)
- {
- (void)arg;
- (void)ignore;
- consdiffmgr_free_all();
- return 1;
- }
- static struct testcase_setup_t setup_diffmgr = {
- consdiffmgr_test_setup,
- consdiffmgr_test_teardown
- };
- // ============================== NS faking functions
- // These functions are for making quick fake consensus objects and
- // strings that are just good enough for consdiff and consdiffmgr.
- static networkstatus_t *
- fake_ns_new(consensus_flavor_t flav, time_t valid_after)
- {
- networkstatus_t *ns = tor_malloc_zero(sizeof(networkstatus_t));
- ns->type = NS_TYPE_CONSENSUS;
- ns->flavor = flav;
- ns->valid_after = valid_after;
- return ns;
- }
- static char *
- fake_ns_body_new(consensus_flavor_t flav, time_t valid_after)
- {
- const char *flavor_string = flav == FLAV_NS ? "" : " microdesc";
- char valid_after_string[ISO_TIME_LEN+1];
- format_iso_time(valid_after_string, valid_after);
- char *random_stuff = crypto_random_hostname(3, 25, "junk ", "");
- char *random_stuff2 = crypto_random_hostname(3, 10, "", "");
- char *consensus;
- tor_asprintf(&consensus,
- "network-status-version 3%s\n"
- "vote-status consensus\n"
- "valid-after %s\n"
- "r name ccccccccccccccccc etc\nsample\n"
- "r name eeeeeeeeeeeeeeeee etc\nbar\n"
- "%s\n"
- "directory-signature hello-there\n"
- "directory-signature %s\n",
- flavor_string,
- valid_after_string,
- random_stuff,
- random_stuff2);
- tor_free(random_stuff);
- tor_free(random_stuff2);
- return consensus;
- }
- // ============================== Cpuworker mocking code
- // These mocking functions and types capture the cpuworker calls
- // so we can inspect them and run them in the main thread.
- static smartlist_t *fake_cpuworker_queue = NULL;
- typedef struct fake_work_queue_ent_t {
- enum workqueue_reply_t (*fn)(void *, void *);
- void (*reply_fn)(void *);
- void *arg;
- } fake_work_queue_ent_t;
- static struct workqueue_entry_s *
- mock_cpuworker_queue_work(workqueue_priority_t prio,
- enum workqueue_reply_t (*fn)(void *, void *),
- void (*reply_fn)(void *),
- void *arg)
- {
- (void) prio;
- if (! fake_cpuworker_queue)
- fake_cpuworker_queue = smartlist_new();
- fake_work_queue_ent_t *ent = tor_malloc_zero(sizeof(*ent));
- ent->fn = fn;
- ent->reply_fn = reply_fn;
- ent->arg = arg;
- smartlist_add(fake_cpuworker_queue, ent);
- return (struct workqueue_entry_s *)ent;
- }
- static int
- mock_cpuworker_run_work(void)
- {
- if (! fake_cpuworker_queue)
- return 0;
- SMARTLIST_FOREACH(fake_cpuworker_queue, fake_work_queue_ent_t *, ent, {
- enum workqueue_reply_t r = ent->fn(NULL, ent->arg);
- if (r != WQ_RPL_REPLY)
- return -1;
- });
- return 0;
- }
- static void
- mock_cpuworker_handle_replies(void)
- {
- if (! fake_cpuworker_queue)
- return;
- SMARTLIST_FOREACH(fake_cpuworker_queue, fake_work_queue_ent_t *, ent, {
- ent->reply_fn(ent->arg);
- tor_free(ent);
- });
- smartlist_free(fake_cpuworker_queue);
- fake_cpuworker_queue = NULL;
- }
- // ============================== Other helpers
- static consdiff_status_t
- lookup_diff_from(consensus_cache_entry_t **out,
- consensus_flavor_t flav,
- const char *str1)
- {
- uint8_t digest[DIGEST256_LEN];
- if (router_get_networkstatus_v3_sha3_as_signed(digest, str1)<0) {
- TT_FAIL(("Unable to compute sha3-as-signed"));
- return CONSDIFF_NOT_FOUND;
- }
- return consdiffmgr_find_diff_from(out, flav,
- DIGEST_SHA3_256, digest, sizeof(digest),
- NO_METHOD);
- }
- static int
- lookup_apply_and_verify_diff(consensus_flavor_t flav,
- const char *str1,
- const char *str2)
- {
- consensus_cache_entry_t *ent = NULL;
- consdiff_status_t status = lookup_diff_from(&ent, flav, str1);
- if (ent == NULL || status != CONSDIFF_AVAILABLE) {
- return -1;
- }
- consensus_cache_entry_incref(ent);
- size_t size;
- char *diff_string = NULL;
- int r = uncompress_or_copy(&diff_string, &size, ent);
- consensus_cache_entry_decref(ent);
- if (diff_string == NULL || r < 0)
- return -1;
- char *applied = consensus_diff_apply(str1, diff_string);
- tor_free(diff_string);
- if (applied == NULL)
- return -1;
- int match = !strcmp(applied, str2);
- tor_free(applied);
- return match ? 0 : -1;
- }
- static void
- cdm_reload(void)
- {
- consdiffmgr_free_all();
- cdm_cache_get();
- consdiffmgr_rescan();
- }
- // ============================== Beginning of tests
- #if 0
- static int got_failure = 0;
- static void
- got_assertion_failure(void)
- {
- ++got_failure;
- }
- /* XXXX This test won't work, because there is currently no way to actually
- * XXXX capture a real assertion failure. */
- static void
- test_consdiffmgr_init_failure(void *arg)
- {
- (void)arg;
- // Capture assertions and bugs.
- /* As in ...test_setup, but do not create the datadir. The missing directory
- * will cause a failure. */
- char *ddir_fname = tor_strdup(get_fname_rnd("datadir_cdm"));
- tor_free(get_options_mutable()->DataDirectory);
- get_options_mutable()->DataDirectory = ddir_fname; // now owns the pointer.
- consdiff_cfg_t consdiff_cfg = { 7200, 300 };
- tor_set_failed_assertion_callback(got_assertion_failure);
- tor_capture_bugs_(1);
- consdiffmgr_configure(&consdiff_cfg); // This should fail.
- tt_int_op(got_failure, OP_EQ, 1);
- const smartlist_t *bugs = tor_get_captured_bug_log_();
- tt_int_op(smartlist_len(bugs), OP_EQ, 1);
- done:
- tor_end_capture_bugs_();
- }
- #endif
- static void
- test_consdiffmgr_sha3_helper(void *arg)
- {
- (void) arg;
- consensus_cache_t *cache = cdm_cache_get(); // violate abstraction barrier
- config_line_t *lines = NULL;
- char *mem_op_hex_tmp = NULL;
- config_line_prepend(&lines, "good-sha",
- "F00DF00DF00DF00DF00DF00DF00DF00D"
- "F00DF00DF00DF00DF00DF00DF00DF00D");
- config_line_prepend(&lines, "short-sha",
- "F00DF00DF00DF00DF00DF00DF00DF00D"
- "F00DF00DF00DF00DF00DF00DF00DF0");
- config_line_prepend(&lines, "long-sha",
- "F00DF00DF00DF00DF00DF00DF00DF00D"
- "F00DF00DF00DF00DF00DF00DF00DF00DF00D");
- config_line_prepend(&lines, "not-sha",
- "F00DF00DF00DF00DF00DF00DF00DF00D"
- "F00DF00DF00DF00DF00DF00DF00DXXXX");
- consensus_cache_entry_t *ent =
- consensus_cache_add(cache, lines, (const uint8_t *)"Hi there", 8);
- uint8_t buf[DIGEST256_LEN];
- tt_int_op(-1, OP_EQ, cdm_entry_get_sha3_value(buf, NULL, "good-sha"));
- tt_int_op(0, OP_EQ, cdm_entry_get_sha3_value(buf, ent, "good-sha"));
- test_memeq_hex(buf, "F00DF00DF00DF00DF00DF00DF00DF00D"
- "F00DF00DF00DF00DF00DF00DF00DF00D");
- tt_int_op(-1, OP_EQ, cdm_entry_get_sha3_value(buf, ent, "missing-sha"));
- tt_int_op(-2, OP_EQ, cdm_entry_get_sha3_value(buf, ent, "short-sha"));
- tt_int_op(-2, OP_EQ, cdm_entry_get_sha3_value(buf, ent, "long-sha"));
- tt_int_op(-2, OP_EQ, cdm_entry_get_sha3_value(buf, ent, "not-sha"));
- done:
- consensus_cache_entry_decref(ent);
- config_free_lines(lines);
- tor_free(mem_op_hex_tmp);
- }
- static void
- test_consdiffmgr_add(void *arg)
- {
- (void) arg;
- time_t now = approx_time();
- char *body = NULL;
- consensus_cache_entry_t *ent = NULL;
- networkstatus_t *ns_tmp = fake_ns_new(FLAV_NS, now);
- const char *dummy = "foo";
- int r = consdiffmgr_add_consensus(dummy, ns_tmp);
- tt_int_op(r, OP_EQ, 0);
- /* If we add it again, it won't work */
- setup_capture_of_logs(LOG_INFO);
- dummy = "bar";
- r = consdiffmgr_add_consensus(dummy, ns_tmp);
- tt_int_op(r, OP_EQ, -1);
- expect_single_log_msg_containing("We already have a copy of that "
- "consensus");
- mock_clean_saved_logs();
- /* But it will work fine if the flavor is different */
- dummy = "baz";
- ns_tmp->flavor = FLAV_MICRODESC;
- r = consdiffmgr_add_consensus(dummy, ns_tmp);
- tt_int_op(r, OP_EQ, 0);
- /* And it will work fine if the time is different */
- dummy = "quux";
- ns_tmp->flavor = FLAV_NS;
- ns_tmp->valid_after = now - 60;
- r = consdiffmgr_add_consensus(dummy, ns_tmp);
- tt_int_op(r, OP_EQ, 0);
- /* If we add one a long long time ago, it will fail. */
- dummy = "xyzzy";
- ns_tmp->valid_after = 86400 * 100; /* A few months into 1970 */
- r = consdiffmgr_add_consensus(dummy, ns_tmp);
- tt_int_op(r, OP_EQ, -1);
- expect_log_msg_containing("it's too old.");
- /* Try looking up a consensuses. */
- ent = cdm_cache_lookup_consensus(FLAV_NS, now-60);
- tt_assert(ent);
- consensus_cache_entry_incref(ent);
- size_t s;
- r = uncompress_or_copy(&body, &s, ent);
- tt_int_op(r, OP_EQ, 0);
- tt_int_op(s, OP_EQ, 4);
- tt_mem_op(body, OP_EQ, "quux", 4);
- /* Try looking up another entry, but fail */
- tt_ptr_op(cdm_cache_lookup_consensus(FLAV_MICRODESC, now - 60), OP_EQ, NULL);
- tt_ptr_op(cdm_cache_lookup_consensus(FLAV_NS, now - 61), OP_EQ, NULL);
- done:
- networkstatus_vote_free(ns_tmp);
- teardown_capture_of_logs();
- consensus_cache_entry_decref(ent);
- tor_free(body);
- }
- static void
- test_consdiffmgr_make_diffs(void *arg)
- {
- (void)arg;
- networkstatus_t *ns = NULL;
- char *ns_body = NULL, *md_ns_body = NULL, *md_ns_body_2 = NULL;
- char *applied = NULL, *diff_text = NULL;
- time_t now = approx_time();
- int r;
- consensus_cache_entry_t *diff = NULL;
- uint8_t md_ns_sha3[DIGEST256_LEN];
- consdiff_status_t diff_status;
- MOCK(cpuworker_queue_work, mock_cpuworker_queue_work);
- // Try rescan with no consensuses: shouldn't crash or queue work.
- consdiffmgr_rescan();
- tt_ptr_op(NULL, OP_EQ, fake_cpuworker_queue);
- // Make two consensuses, 1 hour sec ago.
- ns = fake_ns_new(FLAV_NS, now-3600);
- ns_body = fake_ns_body_new(FLAV_NS, now-3600);
- r = consdiffmgr_add_consensus(ns_body, ns);
- networkstatus_vote_free(ns);
- tor_free(ns_body);
- tt_int_op(r, OP_EQ, 0);
- ns = fake_ns_new(FLAV_MICRODESC, now-3600);
- md_ns_body = fake_ns_body_new(FLAV_MICRODESC, now-3600);
- r = consdiffmgr_add_consensus(md_ns_body, ns);
- router_get_networkstatus_v3_sha3_as_signed(md_ns_sha3, md_ns_body);
- networkstatus_vote_free(ns);
- tt_int_op(r, OP_EQ, 0);
- // No diffs will be generated.
- consdiffmgr_rescan();
- tt_ptr_op(NULL, OP_EQ, fake_cpuworker_queue);
- // Add a MD consensus from 45 minutes ago. This should cause one diff
- // worth of work to get queued.
- ns = fake_ns_new(FLAV_MICRODESC, now-45*60);
- md_ns_body_2 = fake_ns_body_new(FLAV_MICRODESC, now-45*60);
- r = consdiffmgr_add_consensus(md_ns_body_2, ns);
- networkstatus_vote_free(ns);
- tt_int_op(r, OP_EQ, 0);
- consdiffmgr_rescan();
- tt_ptr_op(NULL, OP_NE, fake_cpuworker_queue);
- tt_int_op(1, OP_EQ, smartlist_len(fake_cpuworker_queue));
- diff_status = consdiffmgr_find_diff_from(&diff, FLAV_MICRODESC,
- DIGEST_SHA3_256,
- md_ns_sha3, DIGEST256_LEN,
- NO_METHOD);
- tt_int_op(CONSDIFF_IN_PROGRESS, OP_EQ, diff_status);
- // Now run that process and get the diff.
- r = mock_cpuworker_run_work();
- tt_int_op(r, OP_EQ, 0);
- mock_cpuworker_handle_replies();
- // At this point we should be able to get that diff.
- diff_status = consdiffmgr_find_diff_from(&diff, FLAV_MICRODESC,
- DIGEST_SHA3_256,
- md_ns_sha3, DIGEST256_LEN,
- NO_METHOD);
- tt_int_op(CONSDIFF_AVAILABLE, OP_EQ, diff_status);
- tt_assert(diff);
- /* Make sure applying the diff actually works */
- const uint8_t *diff_body;
- size_t diff_size;
- r = consensus_cache_entry_get_body(diff, &diff_body, &diff_size);
- tt_int_op(r, OP_EQ, 0);
- diff_text = tor_memdup_nulterm(diff_body, diff_size);
- applied = consensus_diff_apply(md_ns_body, diff_text);
- tt_assert(applied);
- tt_str_op(applied, OP_EQ, md_ns_body_2);
- /* Rescan again: no more work to do. */
- consdiffmgr_rescan();
- tt_ptr_op(NULL, OP_EQ, fake_cpuworker_queue);
- done:
- tor_free(md_ns_body);
- tor_free(md_ns_body_2);
- tor_free(diff_text);
- tor_free(applied);
- }
- static void
- test_consdiffmgr_diff_rules(void *arg)
- {
- (void)arg;
- #define N 6
- char *md_body[N], *ns_body[N];
- networkstatus_t *md_ns[N], *ns_ns[N];
- int i;
- MOCK(cpuworker_queue_work, mock_cpuworker_queue_work);
- /* Create a bunch of consensus things at 15-second intervals. */
- time_t start = approx_time() - 120;
- for (i = 0; i < N; ++i) {
- time_t when = start + i * 15;
- md_body[i] = fake_ns_body_new(FLAV_MICRODESC, when);
- ns_body[i] = fake_ns_body_new(FLAV_NS, when);
- md_ns[i] = fake_ns_new(FLAV_MICRODESC, when);
- ns_ns[i] = fake_ns_new(FLAV_NS, when);
- }
- /* For the MD consensuses: add 4 of them, and make sure that
- * diffs are created to one consensus (the most recent) only. */
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(md_body[1], md_ns[1]));
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(md_body[2], md_ns[2]));
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(md_body[3], md_ns[3]));
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(md_body[4], md_ns[4]));
- consdiffmgr_rescan();
- tt_ptr_op(NULL, OP_NE, fake_cpuworker_queue);
- tt_int_op(3, OP_EQ, smartlist_len(fake_cpuworker_queue));
- tt_int_op(0, OP_EQ, mock_cpuworker_run_work());
- mock_cpuworker_handle_replies();
- tt_ptr_op(NULL, OP_EQ, fake_cpuworker_queue);
- /* For the NS consensuses: add 3, generate, and add one older one and
- * make sure that older one is the only one whose diff is generated */
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(ns_body[0], ns_ns[0]));
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(ns_body[1], ns_ns[1]));
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(ns_body[5], ns_ns[5]));
- consdiffmgr_rescan();
- tt_ptr_op(NULL, OP_NE, fake_cpuworker_queue);
- tt_int_op(2, OP_EQ, smartlist_len(fake_cpuworker_queue));
- tt_int_op(0, OP_EQ, mock_cpuworker_run_work());
- mock_cpuworker_handle_replies();
- /* At this point, we should actually have working diffs! */
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_NS, ns_body[0], ns_body[5]));
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_NS, ns_body[1], ns_body[5]));
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[1], md_body[4]));
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[2], md_body[4]));
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[3], md_body[4]));
- /* Self-to-self diff won't be present */
- consensus_cache_entry_t *ent;
- tt_int_op(CONSDIFF_NOT_FOUND, OP_EQ,
- lookup_diff_from(&ent, FLAV_NS, ns_body[5]));
- /* No diff from 2 has been added yet */
- tt_int_op(CONSDIFF_NOT_FOUND, OP_EQ,
- lookup_diff_from(&ent, FLAV_NS, ns_body[2]));
- /* No diff arriving at old things. */
- tt_int_op(-1, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[1], md_body[2]));
- /* No backwards diff */
- tt_int_op(-1, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[4], md_body[3]));
- /* Now, an update: add number 2 and make sure it's the only one whose diff
- * is regenerated. */
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(ns_body[2], ns_ns[2]));
- consdiffmgr_rescan();
- tt_ptr_op(NULL, OP_NE, fake_cpuworker_queue);
- tt_int_op(1, OP_EQ, smartlist_len(fake_cpuworker_queue));
- tt_int_op(0, OP_EQ, mock_cpuworker_run_work());
- mock_cpuworker_handle_replies();
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_NS, ns_body[2], ns_body[5]));
- /* Finally: reload, and make sure that the information is still indexed */
- cdm_reload();
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_NS, ns_body[0], ns_body[5]));
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_NS, ns_body[2], ns_body[5]));
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_NS, ns_body[1], ns_body[5]));
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[1], md_body[4]));
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[2], md_body[4]));
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[3], md_body[4]));
- done:
- for (i = 0; i < N; ++i) {
- tor_free(md_body[i]);
- tor_free(ns_body[i]);
- networkstatus_vote_free(md_ns[i]);
- networkstatus_vote_free(ns_ns[i]);
- }
- UNMOCK(cpuworker_queue_work);
- #undef N
- }
- static void
- test_consdiffmgr_diff_failure(void *arg)
- {
- (void)arg;
- MOCK(cpuworker_queue_work, mock_cpuworker_queue_work);
- /* We're going to make sure that if we have a bogus request where
- * we can't actually compute a diff, the world must not end. */
- networkstatus_t *ns1 = NULL;
- networkstatus_t *ns2 = NULL;
- int r;
- ns1 = fake_ns_new(FLAV_NS, approx_time()-100);
- ns2 = fake_ns_new(FLAV_NS, approx_time()-50);
- r = consdiffmgr_add_consensus("foo bar baz\n", ns1);
- tt_int_op(r, OP_EQ, 0);
- // We refuse to compute a diff to or from a line holding only a single dot.
- // We can add it here, though.
- r = consdiffmgr_add_consensus("foo bar baz\n.\n.\n", ns2);
- tt_int_op(r, OP_EQ, 0);
- consdiffmgr_rescan();
- tt_ptr_op(NULL, OP_NE, fake_cpuworker_queue);
- setup_capture_of_logs(LOG_WARN);
- tt_int_op(1, OP_EQ, smartlist_len(fake_cpuworker_queue));
- tt_int_op(0, OP_EQ, mock_cpuworker_run_work());
- expect_single_log_msg_containing("one of the lines to be added is \".\".");
- mock_clean_saved_logs();
- mock_cpuworker_handle_replies();
- expect_single_log_msg_containing("Worker was unable to compute consensus "
- "diff from ");
- /* Make sure the diff is not present */
- consensus_cache_entry_t *ent;
- tt_int_op(CONSDIFF_NOT_FOUND, OP_EQ,
- lookup_diff_from(&ent, FLAV_NS, "foo bar baz\n"));
- done:
- teardown_capture_of_logs();
- UNMOCK(cpuworker_queue_work);
- networkstatus_vote_free(ns1);
- networkstatus_vote_free(ns2);
- }
- static void
- test_consdiffmgr_diff_pending(void *arg)
- {
- #define N 3
- (void)arg;
- char *md_body[N];
- networkstatus_t *md_ns[N];
- time_t start = approx_time() - 120;
- int i;
- for (i = 0; i < N; ++i) {
- time_t when = start + i * 30;
- md_body[i] = fake_ns_body_new(FLAV_MICRODESC, when);
- md_ns[i] = fake_ns_new(FLAV_MICRODESC, when);
- }
- MOCK(cpuworker_queue_work, mock_cpuworker_queue_work);
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(md_body[1], md_ns[1]));
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(md_body[2], md_ns[2]));
- /* Make a diff */
- consdiffmgr_rescan();
- tt_int_op(1, OP_EQ, smartlist_len(fake_cpuworker_queue));
- /* Look it up. Is it pending? */
- consensus_cache_entry_t *ent = NULL;
- consdiff_status_t diff_status;
- diff_status = lookup_diff_from(&ent, FLAV_MICRODESC, md_body[1]);
- tt_int_op(CONSDIFF_IN_PROGRESS, OP_EQ, diff_status);
- tt_ptr_op(ent, OP_EQ, NULL);
- /* Add another old consensus. only one new diff should launch! */
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(md_body[0], md_ns[0]));
- consdiffmgr_rescan();
- tt_int_op(2, OP_EQ, smartlist_len(fake_cpuworker_queue));
- tt_int_op(0, OP_EQ, mock_cpuworker_run_work());
- mock_cpuworker_handle_replies();
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[0], md_body[2]));
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[1], md_body[2]));
- done:
- UNMOCK(cpuworker_queue_work);
- for (i = 0; i < N; ++i) {
- tor_free(md_body[i]);
- networkstatus_vote_free(md_ns[i]);
- }
- #undef N
- }
- static void
- test_consdiffmgr_cleanup_old(void *arg)
- {
- (void)arg;
- config_line_t *labels = NULL;
- consensus_cache_entry_t *ent = NULL;
- consensus_cache_t *cache = cdm_cache_get(); // violate abstraction barrier
- /* This item will be will be cleanable because it has a valid-after
- * time far in the past. */
- config_line_prepend(&labels, "document-type", "confribble-blarg");
- config_line_prepend(&labels, "consensus-valid-after",
- "1980-10-10T10:10:10");
- ent = consensus_cache_add(cache, labels, (const uint8_t*)"Foo", 3);
- tt_assert(ent);
- consensus_cache_entry_decref(ent);
- setup_capture_of_logs(LOG_DEBUG);
- tt_int_op(1, OP_EQ, consdiffmgr_cleanup());
- expect_log_msg_containing("Deleting entry because its consensus-valid-"
- "after value (1980-10-10T10:10:10) was too old");
- done:
- teardown_capture_of_logs();
- config_free_lines(labels);
- }
- static void
- test_consdiffmgr_cleanup_bad_valid_after(void *arg)
- {
- /* This will seem cleanable, but isn't, because its valid-after time is
- * misformed. */
- (void)arg;
- config_line_t *labels = NULL;
- consensus_cache_entry_t *ent = NULL;
- consensus_cache_t *cache = cdm_cache_get(); // violate abstraction barrier
- config_line_prepend(&labels, "document-type", "consensus");
- config_line_prepend(&labels, "consensus-valid-after",
- "whan that aprille with his shoures soote"); // (~1385?)
- ent = consensus_cache_add(cache, labels, (const uint8_t*)"Foo", 3);
- tt_assert(ent);
- consensus_cache_entry_decref(ent);
- setup_capture_of_logs(LOG_DEBUG);
- tt_int_op(0, OP_EQ, consdiffmgr_cleanup());
- expect_log_msg_containing("Ignoring entry because its consensus-valid-"
- "after value (\"whan that aprille with his "
- "shoures soote\") was unparseable");
- done:
- teardown_capture_of_logs();
- config_free_lines(labels);
- }
- static void
- test_consdiffmgr_cleanup_no_valid_after(void *arg)
- {
- (void)arg;
- config_line_t *labels = NULL;
- consensus_cache_entry_t *ent = NULL;
- consensus_cache_t *cache = cdm_cache_get(); // violate abstraction barrier
- /* This item will be will be uncleanable because it has no recognized
- * valid-after. */
- config_line_prepend(&labels, "document-type", "consensus");
- config_line_prepend(&labels, "confrooble-voolid-oofter",
- "2010-10-10T09:08:07");
- ent = consensus_cache_add(cache, labels, (const uint8_t*)"Foo", 3);
- tt_assert(ent);
- consensus_cache_entry_decref(ent);
- setup_capture_of_logs(LOG_DEBUG);
- tt_int_op(0, OP_EQ, consdiffmgr_cleanup());
- expect_log_msg_containing("Ignoring entry because it had no consensus-"
- "valid-after label");
- done:
- teardown_capture_of_logs();
- config_free_lines(labels);
- }
- static void
- test_consdiffmgr_cleanup_old_diffs(void *arg)
- {
- (void)arg;
- #define N 4
- char *md_body[N];
- networkstatus_t *md_ns[N];
- int i;
- consensus_cache_entry_t *hold_ent = NULL, *ent;
- /* Make sure that the cleanup function removes diffs to the not-most-recent
- * consensus. */
- MOCK(cpuworker_queue_work, mock_cpuworker_queue_work);
- /* Create a bunch of consensus things at 15-second intervals. */
- time_t start = approx_time() - 120;
- for (i = 0; i < N; ++i) {
- time_t when = start + i * 15;
- md_body[i] = fake_ns_body_new(FLAV_MICRODESC, when);
- md_ns[i] = fake_ns_new(FLAV_MICRODESC, when);
- }
- /* add the first 3. */
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(md_body[0], md_ns[0]));
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(md_body[1], md_ns[1]));
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(md_body[2], md_ns[2]));
- /* Make diffs. */
- consdiffmgr_rescan();
- tt_ptr_op(NULL, OP_NE, fake_cpuworker_queue);
- tt_int_op(2, OP_EQ, smartlist_len(fake_cpuworker_queue));
- tt_int_op(0, OP_EQ, mock_cpuworker_run_work());
- mock_cpuworker_handle_replies();
- tt_ptr_op(NULL, OP_EQ, fake_cpuworker_queue);
- /* Nothing is deletable now */
- tt_int_op(0, OP_EQ, consdiffmgr_cleanup());
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[0], md_body[2]));
- tt_int_op(0, OP_EQ,
- lookup_apply_and_verify_diff(FLAV_MICRODESC, md_body[1], md_body[2]));
- tt_int_op(CONSDIFF_AVAILABLE, OP_EQ,
- lookup_diff_from(&hold_ent, FLAV_MICRODESC, md_body[1]));
- consensus_cache_entry_incref(hold_ent); // incref, so it is preserved.
- /* Now add an even-more-recent consensus; this should make all previous
- * diffs deletable, and make delete */
- tt_int_op(0, OP_EQ, consdiffmgr_add_consensus(md_body[3], md_ns[3]));
- tt_int_op(2 * n_diff_compression_methods() +
- (n_consensus_compression_methods() - 1) , OP_EQ,
- consdiffmgr_cleanup());
- tt_int_op(CONSDIFF_NOT_FOUND, OP_EQ,
- lookup_diff_from(&ent, FLAV_MICRODESC, md_body[0]));
- /* This one is marked deletable but still in the hashtable */
- tt_int_op(CONSDIFF_AVAILABLE, OP_EQ,
- lookup_diff_from(&ent, FLAV_MICRODESC, md_body[1]));
- tt_int_op(CONSDIFF_NOT_FOUND, OP_EQ,
- lookup_diff_from(&ent, FLAV_MICRODESC, md_body[2]));
- /* Everything should be valid at this point */
- tt_int_op(0, OP_EQ, consdiffmgr_validate());
- /* And if we recan NOW, we'll purge the hashtable of the entries,
- * and launch attempts to generate new ones */
- consdiffmgr_rescan();
- tt_int_op(CONSDIFF_IN_PROGRESS, OP_EQ,
- lookup_diff_from(&ent, FLAV_MICRODESC, md_body[0]));
- tt_int_op(CONSDIFF_IN_PROGRESS, OP_EQ,
- lookup_diff_from(&ent, FLAV_MICRODESC, md_body[1]));
- tt_int_op(CONSDIFF_IN_PROGRESS, OP_EQ,
- lookup_diff_from(&ent, FLAV_MICRODESC, md_body[2]));
- /* We're still holding on to this, though, so we can still map it! */
- const uint8_t *t1 = NULL;
- size_t s;
- int r = consensus_cache_entry_get_body(hold_ent, &t1, &s);
- tt_int_op(r, OP_EQ, 0);
- tt_assert(t1);
- done:
- for (i = 0; i < N; ++i) {
- tor_free(md_body[i]);
- networkstatus_vote_free(md_ns[i]);
- }
- consensus_cache_entry_decref(hold_ent);
- UNMOCK(cpuworker_queue_work);
- #undef N
- }
- static void
- test_consdiffmgr_validate(void *arg)
- {
- (void)arg;
- config_line_t *lines = NULL;
- consensus_cache_entry_t *ent = NULL;
- consensus_cache_t *cache = cdm_cache_get(); // violate abstraction barrier
- smartlist_t *vals = smartlist_new();
- /* Put these: objects in the cache: one with a good sha3, one with bad sha3,
- * one with a wrong sha3, and one with no sha3. */
- config_line_prepend(&lines, "id", "wrong sha3");
- config_line_prepend(&lines, "sha3-digest",
- "F00DF00DF00DF00DF00DF00DF00DF00D"
- "F00DF00DF00DF00DF00DF00DF00DF00D");
- ent = consensus_cache_add(cache, lines, (const uint8_t *)"Hi there", 8);
- consensus_cache_entry_decref(ent);
- config_free_lines(lines);
- lines = NULL;
- config_line_prepend(&lines, "id", "bad sha3");
- config_line_prepend(&lines, "sha3-digest",
- "now is the winter of our dicotheque");
- ent = consensus_cache_add(cache, lines, (const uint8_t *)"Hi there", 8);
- consensus_cache_entry_decref(ent);
- config_free_lines(lines);
- lines = NULL;
- config_line_prepend(&lines, "id", "no sha3");
- ent = consensus_cache_add(cache, lines, (const uint8_t *)"Hi there", 8);
- consensus_cache_entry_decref(ent);
- config_free_lines(lines);
- lines = NULL;
- config_line_prepend(&lines, "id", "good sha3");
- config_line_prepend(&lines, "sha3-digest",
- "8d8b1998616cd6b4c4055da8d38728dc"
- "93c758d4131a53c7d81aa6337dee1c05");
- ent = consensus_cache_add(cache, lines, (const uint8_t *)"Hi there", 8);
- consensus_cache_entry_decref(ent);
- config_free_lines(lines);
- lines = NULL;
- cdm_reload();
- cache = cdm_cache_get();
- tt_int_op(1, OP_EQ, consdiffmgr_validate());
- consensus_cache_find_all(vals, cache, "id", "good sha3");
- tt_int_op(smartlist_len(vals), OP_EQ, 1);
- smartlist_clear(vals);
- consensus_cache_find_all(vals, cache, "id", "no sha3");
- tt_int_op(smartlist_len(vals), OP_EQ, 1);
- smartlist_clear(vals);
- consensus_cache_find_all(vals, cache, "id", "wrong sha3");
- tt_int_op(smartlist_len(vals), OP_EQ, 0);
- consensus_cache_find_all(vals, cache, "id", "bad sha3");
- tt_int_op(smartlist_len(vals), OP_EQ, 0);
- done:
- smartlist_free(vals);
- }
- #define TEST(name) \
- { #name, test_consdiffmgr_ ## name , TT_FORK, &setup_diffmgr, NULL }
- struct testcase_t consdiffmgr_tests[] = {
- #if 0
- { "init_failure", test_consdiffmgr_init_failure, TT_FORK, NULL, NULL },
- #endif
- TEST(sha3_helper),
- TEST(add),
- TEST(make_diffs),
- TEST(diff_rules),
- TEST(diff_failure),
- TEST(diff_pending),
- TEST(cleanup_old),
- TEST(cleanup_bad_valid_after),
- TEST(cleanup_no_valid_after),
- TEST(cleanup_old_diffs),
- TEST(validate),
- // XXXX Test: non-cacheing cases of replyfn().
- END_OF_TESTCASES
- };
|