test_storagedir.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376
  1. /* Copyright (c) 2017, The Tor Project, Inc. */
  2. /* See LICENSE for licensing information */
  3. #include "or.h"
  4. #include "crypto_rand.h"
  5. #include "storagedir.h"
  6. #include "test.h"
  7. #ifdef HAVE_UTIME_H
  8. #include <utime.h>
  9. #endif
  10. static void
  11. test_storagedir_empty(void *arg)
  12. {
  13. char *dirname = tor_strdup(get_fname_rnd("store_dir"));
  14. storage_dir_t *d = NULL;
  15. (void)arg;
  16. tt_int_op(FN_NOENT, OP_EQ, file_status(dirname));
  17. d = storage_dir_new(dirname, 10);
  18. tt_assert(d);
  19. tt_int_op(FN_DIR, OP_EQ, file_status(dirname));
  20. tt_int_op(0, OP_EQ, smartlist_len(storage_dir_list(d)));
  21. tt_u64_op(0, OP_EQ, storage_dir_get_usage(d));
  22. storage_dir_free(d);
  23. d = storage_dir_new(dirname, 10);
  24. tt_assert(d);
  25. tt_int_op(FN_DIR, OP_EQ, file_status(dirname));
  26. tt_int_op(0, OP_EQ, smartlist_len(storage_dir_list(d)));
  27. tt_u64_op(0, OP_EQ, storage_dir_get_usage(d));
  28. done:
  29. storage_dir_free(d);
  30. tor_free(dirname);
  31. }
  32. static void
  33. test_storagedir_basic(void *arg)
  34. {
  35. char *dirname = tor_strdup(get_fname_rnd("store_dir"));
  36. storage_dir_t *d = NULL;
  37. uint8_t *junk = NULL, *bytes = NULL;
  38. const size_t junklen = 1024;
  39. char *fname1 = NULL, *fname2 = NULL;
  40. const char hello_str[] = "then what are we but cold, alone ... ?";
  41. tor_mmap_t *mapping = NULL;
  42. (void)arg;
  43. junk = tor_malloc(junklen);
  44. crypto_rand((void*)junk, junklen);
  45. d = storage_dir_new(dirname, 10);
  46. tt_assert(d);
  47. tt_u64_op(0, OP_EQ, storage_dir_get_usage(d));
  48. int r;
  49. r = storage_dir_save_string_to_file(d, hello_str, 1, &fname1);
  50. tt_int_op(r, OP_EQ, 0);
  51. tt_ptr_op(fname1, OP_NE, NULL);
  52. tt_u64_op(strlen(hello_str), OP_EQ, storage_dir_get_usage(d));
  53. r = storage_dir_save_bytes_to_file(d, junk, junklen, 1, &fname2);
  54. tt_int_op(r, OP_EQ, 0);
  55. tt_ptr_op(fname2, OP_NE, NULL);
  56. tt_str_op(fname1, OP_NE, fname2);
  57. tt_int_op(2, OP_EQ, smartlist_len(storage_dir_list(d)));
  58. tt_u64_op(junklen + strlen(hello_str), OP_EQ, storage_dir_get_usage(d));
  59. tt_assert(smartlist_contains_string(storage_dir_list(d), fname1));
  60. tt_assert(smartlist_contains_string(storage_dir_list(d), fname2));
  61. storage_dir_free(d);
  62. d = storage_dir_new(dirname, 10);
  63. tt_assert(d);
  64. tt_int_op(2, OP_EQ, smartlist_len(storage_dir_list(d)));
  65. tt_u64_op(junklen + strlen(hello_str), OP_EQ, storage_dir_get_usage(d));
  66. tt_assert(smartlist_contains_string(storage_dir_list(d), fname1));
  67. tt_assert(smartlist_contains_string(storage_dir_list(d), fname2));
  68. size_t n;
  69. bytes = storage_dir_read(d, fname2, 1, &n);
  70. tt_assert(bytes);
  71. tt_u64_op(n, OP_EQ, junklen);
  72. tt_mem_op(bytes, OP_EQ, junk, junklen);
  73. mapping = storage_dir_map(d, fname1);
  74. tt_assert(mapping);
  75. tt_u64_op(mapping->size, OP_EQ, strlen(hello_str));
  76. tt_mem_op(mapping->data, OP_EQ, hello_str, strlen(hello_str));
  77. done:
  78. tor_free(dirname);
  79. tor_free(junk);
  80. tor_free(bytes);
  81. tor_munmap_file(mapping);
  82. storage_dir_free(d);
  83. tor_free(fname1);
  84. tor_free(fname2);
  85. }
  86. static void
  87. test_storagedir_deletion(void *arg)
  88. {
  89. (void)arg;
  90. char *dirname = tor_strdup(get_fname_rnd("store_dir"));
  91. storage_dir_t *d = NULL;
  92. char *fn1 = NULL, *fn2 = NULL;
  93. char *bytes = NULL;
  94. int r;
  95. const char str1[] = "There are nine and sixty ways to disguise communiques";
  96. const char str2[] = "And rather more than one of them is right";
  97. // Make sure the directory is there. */
  98. d = storage_dir_new(dirname, 10);
  99. storage_dir_free(d);
  100. d = NULL;
  101. tor_asprintf(&fn1, "%s/1007", dirname);
  102. r = write_str_to_file(fn1, str1, 0);
  103. tt_int_op(r, OP_EQ, 0);
  104. tor_asprintf(&fn2, "%s/1003.tmp", dirname);
  105. r = write_str_to_file(fn2, str2, 0);
  106. tt_int_op(r, OP_EQ, 0);
  107. // The tempfile should be deleted the next time we list the directory.
  108. d = storage_dir_new(dirname, 10);
  109. tt_int_op(1, OP_EQ, smartlist_len(storage_dir_list(d)));
  110. tt_u64_op(strlen(str1), OP_EQ, storage_dir_get_usage(d));
  111. tt_int_op(FN_FILE, OP_EQ, file_status(fn1));
  112. tt_int_op(FN_NOENT, OP_EQ, file_status(fn2));
  113. bytes = (char*) storage_dir_read(d, "1007", 1, NULL);
  114. tt_str_op(bytes, OP_EQ, str1);
  115. // Should have no effect; file already gone.
  116. storage_dir_remove_file(d, "1003.tmp");
  117. tt_int_op(1, OP_EQ, smartlist_len(storage_dir_list(d)));
  118. tt_u64_op(strlen(str1), OP_EQ, storage_dir_get_usage(d));
  119. // Actually remove a file.
  120. storage_dir_remove_file(d, "1007");
  121. tt_int_op(FN_NOENT, OP_EQ, file_status(fn1));
  122. tt_int_op(0, OP_EQ, smartlist_len(storage_dir_list(d)));
  123. tt_u64_op(0, OP_EQ, storage_dir_get_usage(d));
  124. done:
  125. tor_free(dirname);
  126. tor_free(fn1);
  127. tor_free(fn2);
  128. storage_dir_free(d);
  129. tor_free(bytes);
  130. }
  131. static void
  132. test_storagedir_full(void *arg)
  133. {
  134. (void)arg;
  135. char *dirname = tor_strdup(get_fname_rnd("store_dir"));
  136. storage_dir_t *d = NULL;
  137. const char str[] = "enemies of the peephole";
  138. int r;
  139. d = storage_dir_new(dirname, 3);
  140. tt_assert(d);
  141. r = storage_dir_save_string_to_file(d, str, 1, NULL);
  142. tt_int_op(r, OP_EQ, 0);
  143. r = storage_dir_save_string_to_file(d, str, 1, NULL);
  144. tt_int_op(r, OP_EQ, 0);
  145. r = storage_dir_save_string_to_file(d, str, 1, NULL);
  146. tt_int_op(r, OP_EQ, 0);
  147. // These should fail!
  148. r = storage_dir_save_string_to_file(d, str, 1, NULL);
  149. tt_int_op(r, OP_EQ, -1);
  150. r = storage_dir_save_string_to_file(d, str, 1, NULL);
  151. tt_int_op(r, OP_EQ, -1);
  152. tt_u64_op(strlen(str) * 3, OP_EQ, storage_dir_get_usage(d));
  153. done:
  154. tor_free(dirname);
  155. storage_dir_free(d);
  156. }
  157. static void
  158. test_storagedir_cleaning(void *arg)
  159. {
  160. (void)arg;
  161. char *dirname = tor_strdup(get_fname_rnd("store_dir"));
  162. storage_dir_t *d = NULL;
  163. const char str[] =
  164. "On a mountain halfway between Reno and Rome / "
  165. "We have a machine in a plexiglass dome / "
  166. "Which listens and looks into everyone's home."
  167. " -- Dr. Seuss";
  168. char *fns[8];
  169. int r, i;
  170. memset(fns, 0, sizeof(fns));
  171. d = storage_dir_new(dirname, 10);
  172. tt_assert(d);
  173. for (i = 0; i < 8; ++i) {
  174. r = storage_dir_save_string_to_file(d, str+i*2, 1, &fns[i]);
  175. tt_int_op(r, OP_EQ, 0);
  176. }
  177. /* Now we're going to make sure all the files have distinct mtimes. */
  178. time_t now = time(NULL);
  179. struct utimbuf ub;
  180. ub.actime = now;
  181. ub.modtime = now - 1000;
  182. for (i = 0; i < 8; ++i) {
  183. char *f = NULL;
  184. tor_asprintf(&f, "%s/%s", dirname, fns[i]);
  185. r = utime(f, &ub);
  186. tor_free(f);
  187. tt_int_op(r, OP_EQ, 0);
  188. ub.modtime += 5;
  189. }
  190. const uint64_t usage_orig = storage_dir_get_usage(d);
  191. /* No changes needed if we are already under target. */
  192. storage_dir_shrink(d, 1024*1024, 0);
  193. tt_u64_op(usage_orig, OP_EQ, storage_dir_get_usage(d));
  194. /* Get rid of at least one byte. This will delete fns[0]. */
  195. storage_dir_shrink(d, usage_orig - 1, 0);
  196. tt_u64_op(usage_orig, OP_GT, storage_dir_get_usage(d));
  197. tt_u64_op(usage_orig - strlen(str), OP_EQ, storage_dir_get_usage(d));
  198. /* Get rid of at least two files. This will delete fns[1] and fns[2]. */
  199. storage_dir_shrink(d, 1024*1024, 2);
  200. tt_u64_op(usage_orig - strlen(str)*3 + 6, OP_EQ, storage_dir_get_usage(d));
  201. /* Get rid of everything. */
  202. storage_dir_remove_all(d);
  203. tt_u64_op(0, OP_EQ, storage_dir_get_usage(d));
  204. done:
  205. tor_free(dirname);
  206. storage_dir_free(d);
  207. for (i = 0; i < 8; ++i) {
  208. tor_free(fns[i]);
  209. }
  210. }
  211. static void
  212. test_storagedir_save_labeled(void *arg)
  213. {
  214. (void)arg;
  215. char *dirname = tor_strdup(get_fname_rnd("store_dir"));
  216. storage_dir_t *d = NULL;
  217. uint8_t *inp = tor_malloc_zero(8192);
  218. config_line_t *labels = NULL;
  219. char *fname = NULL;
  220. uint8_t *saved = NULL;
  221. d = storage_dir_new(dirname, 10);
  222. tt_assert(d);
  223. crypto_rand((char *)inp, 8192);
  224. config_line_append(&labels, "Foo", "bar baz");
  225. config_line_append(&labels, "quux", "quuzXxz");
  226. const char expected[] =
  227. "Foo bar baz\n"
  228. "quux quuzXxz\n";
  229. int r = storage_dir_save_labeled_to_file(d, labels, inp, 8192, &fname);
  230. tt_int_op(r, OP_EQ, 0);
  231. size_t n;
  232. saved = storage_dir_read(d, fname, 1, &n);
  233. tt_assert(memchr(saved, '\0', n));
  234. tt_str_op((char*)saved, OP_EQ, expected); /* NUL guarantees strcmp works */
  235. tt_mem_op(saved+strlen(expected)+1, OP_EQ, inp, 8192);
  236. done:
  237. storage_dir_free(d);
  238. tor_free(dirname);
  239. tor_free(inp);
  240. tor_free(fname);
  241. config_free_lines(labels);
  242. tor_free(saved);
  243. }
  244. static void
  245. test_storagedir_read_labeled(void *arg)
  246. {
  247. (void)arg;
  248. char *dirname = tor_strdup(get_fname_rnd("store_dir"));
  249. storage_dir_t *d = NULL;
  250. uint8_t *inp = tor_malloc_zero(8192);
  251. config_line_t *labels = NULL, *labels2 = NULL;
  252. char *fname = NULL;
  253. tor_mmap_t *map = NULL;
  254. uint8_t *as_read = NULL;
  255. d = storage_dir_new(dirname, 10);
  256. tt_assert(d);
  257. tor_snprintf((char*)inp, 8192,
  258. "Hello world\n"
  259. "This is a test\n"
  260. "Yadda yadda.\n");
  261. size_t bodylen = 8192 - strlen((char*)inp) - 1;
  262. crypto_rand((char *)inp+strlen((char*)inp)+1, bodylen);
  263. int r = storage_dir_save_bytes_to_file(d, inp, 8192, 1, &fname);
  264. tt_int_op(r, OP_EQ, 0);
  265. /* Try mapping */
  266. const uint8_t *datap = NULL;
  267. size_t sz = 0;
  268. map = storage_dir_map_labeled(d, fname, &labels, &datap, &sz);
  269. tt_assert(map);
  270. tt_assert(datap);
  271. tt_u64_op(sz, OP_EQ, bodylen);
  272. tt_mem_op(datap, OP_EQ, inp+strlen((char*)inp)+1, bodylen);
  273. tt_assert(labels);
  274. tt_str_op(labels->key, OP_EQ, "Hello");
  275. tt_str_op(labels->value, OP_EQ, "world");
  276. tt_assert(labels->next);
  277. tt_str_op(labels->next->key, OP_EQ, "This");
  278. tt_str_op(labels->next->value, OP_EQ, "is a test");
  279. tt_assert(labels->next->next);
  280. tt_str_op(labels->next->next->key, OP_EQ, "Yadda");
  281. tt_str_op(labels->next->next->value, OP_EQ, "yadda.");
  282. tt_ptr_op(labels->next->next->next, OP_EQ, NULL);
  283. /* Try reading this time. */
  284. sz = 0;
  285. as_read = storage_dir_read_labeled(d, fname, &labels2, &sz);
  286. tt_assert(as_read);
  287. tt_u64_op(sz, OP_EQ, bodylen);
  288. tt_mem_op(as_read, OP_EQ, inp+strlen((char*)inp)+1, bodylen);
  289. tt_assert(config_lines_eq(labels, labels2));
  290. done:
  291. storage_dir_free(d);
  292. tor_free(dirname);
  293. tor_free(inp);
  294. tor_free(fname);
  295. config_free_lines(labels);
  296. config_free_lines(labels2);
  297. tor_munmap_file(map);
  298. tor_free(as_read);
  299. }
  300. #define ENT(name) \
  301. { #name, test_storagedir_ ## name, TT_FORK, NULL, NULL }
  302. struct testcase_t storagedir_tests[] = {
  303. ENT(empty),
  304. ENT(basic),
  305. ENT(deletion),
  306. ENT(full),
  307. ENT(cleaning),
  308. ENT(save_labeled),
  309. ENT(read_labeled),
  310. END_OF_TESTCASES
  311. };