test_entry.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <zip.h>
  4. #include "minunit.h"
  5. #if defined(_WIN32) || defined(_WIN64)
  6. #define MKTEMP _mktemp
  7. #define UNLINK _unlink
  8. #else
  9. #define MKTEMP mkstemp
  10. #define UNLINK unlink
  11. #endif
  12. #define MZ_ZIP64_MAX_CENTRAL_EXTRA_FIELD_SIZE \
  13. (sizeof(unsigned short) * 2 + sizeof(unsigned long long) * 3)
  14. #define MZ_ZIP_LOCAL_DIR_HEADER_SIZE 30
  15. static char ZIPNAME[L_tmpnam + 1] = {0};
  16. #define CRC32DATA1 2220805626
  17. #define TESTDATA1 "Some test data 1...\0"
  18. #define TESTDATA2 "Some test data 2...\0"
  19. #define CRC32DATA2 2532008468
  20. static int total_entries = 0;
  21. void test_setup(void) {
  22. strncpy(ZIPNAME, "z-XXXXXX\0", L_tmpnam);
  23. MKTEMP(ZIPNAME);
  24. struct zip_t *zip = zip_open(ZIPNAME, ZIP_DEFAULT_COMPRESSION_LEVEL, 'w');
  25. zip_entry_open(zip, "test/test-1.txt");
  26. zip_entry_write(zip, TESTDATA1, strlen(TESTDATA1));
  27. zip_entry_close(zip);
  28. ++total_entries;
  29. zip_entry_open(zip, "test\\test-2.txt");
  30. zip_entry_write(zip, TESTDATA2, strlen(TESTDATA2));
  31. zip_entry_close(zip);
  32. ++total_entries;
  33. zip_entry_open(zip, "test\\empty/");
  34. zip_entry_close(zip);
  35. ++total_entries;
  36. zip_entry_open(zip, "empty/");
  37. zip_entry_close(zip);
  38. ++total_entries;
  39. zip_entry_open(zip, "dotfiles/.test");
  40. zip_entry_write(zip, TESTDATA2, strlen(TESTDATA2));
  41. zip_entry_close(zip);
  42. ++total_entries;
  43. zip_entry_open(zip, "delete.me");
  44. zip_entry_write(zip, TESTDATA1, strlen(TESTDATA1));
  45. zip_entry_close(zip);
  46. ++total_entries;
  47. zip_entry_open(zip, "_");
  48. zip_entry_write(zip, TESTDATA2, strlen(TESTDATA2));
  49. zip_entry_close(zip);
  50. ++total_entries;
  51. zip_entry_open(zip, "delete/file.1");
  52. zip_entry_write(zip, TESTDATA1, strlen(TESTDATA1));
  53. zip_entry_close(zip);
  54. ++total_entries;
  55. zip_entry_open(zip, "delete/file.2");
  56. zip_entry_write(zip, TESTDATA2, strlen(TESTDATA2));
  57. zip_entry_close(zip);
  58. ++total_entries;
  59. zip_entry_open(zip, "deleteme/file.3");
  60. zip_entry_write(zip, TESTDATA1, strlen(TESTDATA1));
  61. zip_entry_close(zip);
  62. ++total_entries;
  63. zip_entry_open(zip, "delete/file.4");
  64. zip_entry_write(zip, TESTDATA2, strlen(TESTDATA2));
  65. zip_entry_close(zip);
  66. ++total_entries;
  67. zip_close(zip);
  68. }
  69. void test_teardown(void) {
  70. total_entries = 0;
  71. UNLINK(ZIPNAME);
  72. }
  73. MU_TEST(test_entry_name) {
  74. struct zip_t *zip = zip_open(ZIPNAME, 0, 'r');
  75. mu_check(zip != NULL);
  76. mu_check(zip_entry_name(zip) == NULL);
  77. mu_assert_int_eq(0, zip_entry_open(zip, "test\\test-1.txt"));
  78. mu_check(NULL != zip_entry_name(zip));
  79. mu_assert_int_eq(0, strcmp(zip_entry_name(zip), "test/test-1.txt"));
  80. mu_assert_int_eq(strlen(TESTDATA1), zip_entry_size(zip));
  81. mu_check(CRC32DATA1 == zip_entry_crc32(zip));
  82. mu_assert_int_eq(0, zip_entry_index(zip));
  83. mu_assert_int_eq(0, zip_entry_close(zip));
  84. mu_assert_int_eq(0, zip_entry_open(zip, "test/test-2.txt"));
  85. mu_check(NULL != zip_entry_name(zip));
  86. mu_assert_int_eq(0, strcmp(zip_entry_name(zip), "test/test-2.txt"));
  87. mu_assert_int_eq(strlen(TESTDATA2), zip_entry_size(zip));
  88. mu_check(CRC32DATA2 == zip_entry_crc32(zip));
  89. mu_assert_int_eq(1, zip_entry_index(zip));
  90. mu_assert_int_eq(0, zip_entry_close(zip));
  91. zip_close(zip);
  92. }
  93. MU_TEST(test_entry_opencasesensitive) {
  94. struct zip_t *zip = zip_open(ZIPNAME, 0, 'r');
  95. mu_check(zip != NULL);
  96. mu_check(zip_entry_name(zip) == NULL);
  97. mu_assert_int_eq(0, zip_entry_open(zip, "test/TEST-1.TXT"));
  98. mu_check(NULL != zip_entry_name(zip));
  99. mu_assert_int_eq(0, zip_entry_close(zip));
  100. mu_assert_int_eq(ZIP_ENOENT,
  101. zip_entry_opencasesensitive(zip, "test/TEST-1.TXT"));
  102. zip_close(zip);
  103. }
  104. MU_TEST(test_entry_index) {
  105. struct zip_t *zip = zip_open(ZIPNAME, 0, 'r');
  106. mu_check(zip != NULL);
  107. mu_assert_int_eq(0, zip_entry_open(zip, "test\\test-1.txt"));
  108. mu_assert_int_eq(0, zip_entry_index(zip));
  109. mu_assert_int_eq(0, strcmp(zip_entry_name(zip), "test/test-1.txt"));
  110. mu_assert_int_eq(strlen(TESTDATA1), zip_entry_size(zip));
  111. mu_check(CRC32DATA1 == zip_entry_crc32(zip));
  112. mu_assert_int_eq(0, zip_entry_close(zip));
  113. mu_assert_int_eq(0, zip_entry_open(zip, "test/test-2.txt"));
  114. mu_assert_int_eq(1, zip_entry_index(zip));
  115. mu_assert_int_eq(0, strcmp(zip_entry_name(zip), "test/test-2.txt"));
  116. mu_assert_int_eq(strlen(TESTDATA2), zip_entry_size(zip));
  117. mu_check(CRC32DATA2 == zip_entry_crc32(zip));
  118. mu_assert_int_eq(0, zip_entry_close(zip));
  119. zip_close(zip);
  120. }
  121. MU_TEST(test_entry_openbyindex) {
  122. struct zip_t *zip = zip_open(ZIPNAME, 0, 'r');
  123. mu_check(zip != NULL);
  124. mu_assert_int_eq(0, zip_entry_openbyindex(zip, 1));
  125. mu_assert_int_eq(1, zip_entry_index(zip));
  126. mu_assert_int_eq(strlen(TESTDATA2), zip_entry_size(zip));
  127. mu_check(CRC32DATA2 == zip_entry_crc32(zip));
  128. mu_assert_int_eq(0, strcmp(zip_entry_name(zip), "test/test-2.txt"));
  129. mu_assert_int_eq(0, zip_entry_close(zip));
  130. mu_assert_int_eq(0, zip_entry_openbyindex(zip, 0));
  131. mu_assert_int_eq(0, zip_entry_index(zip));
  132. mu_assert_int_eq(strlen(TESTDATA1), zip_entry_size(zip));
  133. mu_check(CRC32DATA1 == zip_entry_crc32(zip));
  134. mu_assert_int_eq(0, strcmp(zip_entry_name(zip), "test/test-1.txt"));
  135. mu_assert_int_eq(0, zip_entry_close(zip));
  136. zip_close(zip);
  137. }
  138. MU_TEST(test_entry_read) {
  139. char *bufencode1 = NULL;
  140. char *bufencode2 = NULL;
  141. char *buf = NULL;
  142. size_t bufsize;
  143. struct zip_t *zip =
  144. zip_stream_open(NULL, 0, ZIP_DEFAULT_COMPRESSION_LEVEL, 'w');
  145. mu_check(zip != NULL);
  146. mu_assert_int_eq(0, zip_entry_open(zip, "test/test-1.txt"));
  147. mu_assert_int_eq(0, zip_entry_write(zip, TESTDATA1, strlen(TESTDATA1)));
  148. mu_assert_int_eq(0, zip_entry_close(zip));
  149. ssize_t n = zip_stream_copy(zip, (void **)&bufencode1, NULL);
  150. zip_stream_copy(zip, (void **)&bufencode2, &bufsize);
  151. mu_assert_int_eq(0, strncmp(bufencode1, bufencode2, bufsize));
  152. zip_stream_close(zip);
  153. struct zip_t *zipstream = zip_stream_open(bufencode1, n, 0, 'r');
  154. mu_check(zipstream != NULL);
  155. mu_assert_int_eq(0, zip_entry_open(zipstream, "test/test-1.txt"));
  156. n = zip_entry_read(zipstream, (void **)&buf, NULL);
  157. mu_assert_int_eq(0, strncmp(buf, TESTDATA1, (size_t)n));
  158. mu_assert_int_eq(0, zip_entry_close(zipstream));
  159. zip_stream_close(zipstream);
  160. free(buf);
  161. free(bufencode1);
  162. free(bufencode2);
  163. }
  164. MU_TEST(test_list_entries) {
  165. struct zip_t *zip = zip_open(ZIPNAME, 0, 'r');
  166. mu_check(zip != NULL);
  167. int i = 0, n = zip_entries_total(zip);
  168. for (; i < n; ++i) {
  169. mu_assert_int_eq(0, zip_entry_openbyindex(zip, i));
  170. fprintf(stdout, "[%d]: %s", i, zip_entry_name(zip));
  171. if (zip_entry_isdir(zip)) {
  172. fprintf(stdout, " (DIR)");
  173. }
  174. fprintf(stdout, "\n");
  175. mu_assert_int_eq(0, zip_entry_close(zip));
  176. }
  177. zip_close(zip);
  178. }
  179. MU_TEST(test_entries_deletebyindex) {
  180. size_t entries[] = {5, 6, 7, 9, 8};
  181. struct zip_t *zip = zip_open(ZIPNAME, 0, 'd');
  182. mu_check(zip != NULL);
  183. mu_assert_int_eq(5, zip_entries_deletebyindex(zip, entries, 5));
  184. zip_close(zip);
  185. zip = zip_open(ZIPNAME, 0, 'r');
  186. mu_check(zip != NULL);
  187. mu_assert_int_eq(ZIP_ENOENT, zip_entry_open(zip, "delete.me"));
  188. mu_assert_int_eq(0, zip_entry_close(zip));
  189. fprintf(stdout, "delete.me: %s\n", zip_strerror(ZIP_ENOENT));
  190. mu_assert_int_eq(ZIP_ENOENT, zip_entry_open(zip, "_"));
  191. mu_assert_int_eq(0, zip_entry_close(zip));
  192. fprintf(stdout, "_: %s\n", zip_strerror(ZIP_ENOENT));
  193. mu_assert_int_eq(ZIP_ENOENT, zip_entry_open(zip, "delete/file.1"));
  194. mu_assert_int_eq(0, zip_entry_close(zip));
  195. fprintf(stdout, "delete/file.1: %s\n", zip_strerror(ZIP_ENOENT));
  196. mu_assert_int_eq(ZIP_ENOENT, zip_entry_open(zip, "deleteme/file.3"));
  197. mu_assert_int_eq(0, zip_entry_close(zip));
  198. fprintf(stdout, "delete/file.3: %s\n", zip_strerror(ZIP_ENOENT));
  199. mu_assert_int_eq(ZIP_ENOENT, zip_entry_open(zip, "delete/file.2"));
  200. mu_assert_int_eq(0, zip_entry_close(zip));
  201. fprintf(stdout, "delete/file.2: %s\n", zip_strerror(ZIP_ENOENT));
  202. mu_assert_int_eq(total_entries - 5, zip_entries_total(zip));
  203. mu_assert_int_eq(0, zip_entry_open(zip, "delete/file.4"));
  204. size_t buftmp = 0;
  205. char *buf = NULL;
  206. ssize_t bufsize = zip_entry_read(zip, (void **)&buf, &buftmp);
  207. mu_assert_int_eq(bufsize, strlen(TESTDATA2));
  208. mu_assert_int_eq((size_t)bufsize, buftmp);
  209. mu_assert_int_eq(0, strncmp(buf, TESTDATA2, bufsize));
  210. mu_assert_int_eq(0, zip_entry_close(zip));
  211. free(buf);
  212. buf = NULL;
  213. zip_close(zip);
  214. }
  215. MU_TEST(test_entries_deleteinvalid) {
  216. size_t entries[] = {111, 222, 333, 444};
  217. struct zip_t *zip = zip_open(ZIPNAME, 0, 'd');
  218. mu_check(zip != NULL);
  219. mu_assert_int_eq(0, zip_entries_deletebyindex(zip, entries, 4));
  220. zip_close(zip);
  221. zip = zip_open(ZIPNAME, 0, 'r');
  222. mu_check(zip != NULL);
  223. mu_assert_int_eq(0, zip_entry_open(zip, "delete.me"));
  224. mu_assert_int_eq(0, zip_entry_close(zip));
  225. mu_assert_int_eq(0, zip_entry_open(zip, "_"));
  226. mu_assert_int_eq(0, zip_entry_close(zip));
  227. mu_assert_int_eq(0, zip_entry_open(zip, "delete/file.1"));
  228. mu_assert_int_eq(0, zip_entry_close(zip));
  229. mu_assert_int_eq(0, zip_entry_open(zip, "deleteme/file.3"));
  230. mu_assert_int_eq(0, zip_entry_close(zip));
  231. mu_assert_int_eq(0, zip_entry_open(zip, "delete/file.2"));
  232. mu_assert_int_eq(0, zip_entry_close(zip));
  233. mu_assert_int_eq(total_entries, zip_entries_total(zip));
  234. mu_assert_int_eq(0, zip_entry_open(zip, "delete/file.4"));
  235. size_t buftmp = 0;
  236. char *buf = NULL;
  237. ssize_t bufsize = zip_entry_read(zip, (void **)&buf, &buftmp);
  238. mu_assert_int_eq(bufsize, strlen(TESTDATA2));
  239. mu_assert_int_eq((size_t)bufsize, buftmp);
  240. mu_assert_int_eq(0, strncmp(buf, TESTDATA2, bufsize));
  241. mu_assert_int_eq(0, zip_entry_close(zip));
  242. free(buf);
  243. buf = NULL;
  244. zip_close(zip);
  245. }
  246. MU_TEST(test_entries_delete) {
  247. char *entries[] = {"delete.me", "_", "delete/file.1", "deleteme/file.3",
  248. "delete/file.2"};
  249. struct zip_t *zip = zip_open(ZIPNAME, 0, 'd');
  250. mu_check(zip != NULL);
  251. mu_assert_int_eq(5, zip_entries_delete(zip, entries, 5));
  252. zip_close(zip);
  253. zip = zip_open(ZIPNAME, 0, 'r');
  254. mu_check(zip != NULL);
  255. mu_assert_int_eq(ZIP_ENOENT, zip_entry_open(zip, "delete.me"));
  256. mu_assert_int_eq(0, zip_entry_close(zip));
  257. fprintf(stdout, "delete.me: %s\n", zip_strerror(ZIP_ENOENT));
  258. mu_assert_int_eq(ZIP_ENOENT, zip_entry_open(zip, "_"));
  259. mu_assert_int_eq(0, zip_entry_close(zip));
  260. fprintf(stdout, "_: %s\n", zip_strerror(ZIP_ENOENT));
  261. mu_assert_int_eq(ZIP_ENOENT, zip_entry_open(zip, "delete/file.1"));
  262. mu_assert_int_eq(0, zip_entry_close(zip));
  263. fprintf(stdout, "delete/file.1: %s\n", zip_strerror(ZIP_ENOENT));
  264. mu_assert_int_eq(ZIP_ENOENT, zip_entry_open(zip, "deleteme/file.3"));
  265. mu_assert_int_eq(0, zip_entry_close(zip));
  266. fprintf(stdout, "delete/file.3: %s\n", zip_strerror(ZIP_ENOENT));
  267. mu_assert_int_eq(ZIP_ENOENT, zip_entry_open(zip, "delete/file.2"));
  268. mu_assert_int_eq(0, zip_entry_close(zip));
  269. fprintf(stdout, "delete/file.2: %s\n", zip_strerror(ZIP_ENOENT));
  270. mu_assert_int_eq(total_entries - 5, zip_entries_total(zip));
  271. mu_assert_int_eq(0, zip_entry_open(zip, "delete/file.4"));
  272. size_t buftmp = 0;
  273. char *buf = NULL;
  274. ssize_t bufsize = zip_entry_read(zip, (void **)&buf, &buftmp);
  275. mu_assert_int_eq(bufsize, strlen(TESTDATA2));
  276. mu_assert_int_eq((size_t)bufsize, buftmp);
  277. mu_assert_int_eq(0, strncmp(buf, TESTDATA2, bufsize));
  278. mu_assert_int_eq(0, zip_entry_close(zip));
  279. free(buf);
  280. buf = NULL;
  281. zip_close(zip);
  282. }
  283. MU_TEST(test_entry_offset) {
  284. struct zip_t *zip = zip_open(ZIPNAME, 0, 'r');
  285. mu_check(zip != NULL);
  286. unsigned long long off = 0ULL;
  287. int i = 0, n = zip_entries_total(zip);
  288. for (; i < n; i++) {
  289. mu_assert_int_eq(0, zip_entry_openbyindex(zip, i));
  290. mu_assert_int_eq(i, zip_entry_index(zip));
  291. mu_assert_int_eq(off, zip_entry_header_offset(zip));
  292. off = zip_entry_header_offset(zip) + MZ_ZIP_LOCAL_DIR_HEADER_SIZE +
  293. strlen(zip_entry_name(zip)) + MZ_ZIP64_MAX_CENTRAL_EXTRA_FIELD_SIZE +
  294. zip_entry_comp_size(zip);
  295. fprintf(stdout, "\n[%d: %s]: header: %llu, dir: %llu, size: %llu (%llu)\n",
  296. i, zip_entry_name(zip), zip_entry_header_offset(zip),
  297. zip_entry_dir_offset(zip), zip_entry_comp_size(zip), off);
  298. mu_assert_int_eq(0, zip_entry_close(zip));
  299. }
  300. zip_close(zip);
  301. }
  302. MU_TEST_SUITE(test_entry_suite) {
  303. MU_SUITE_CONFIGURE(&test_setup, &test_teardown);
  304. MU_RUN_TEST(test_entry_name);
  305. MU_RUN_TEST(test_entry_opencasesensitive);
  306. MU_RUN_TEST(test_entry_index);
  307. MU_RUN_TEST(test_entry_openbyindex);
  308. MU_RUN_TEST(test_entry_read);
  309. MU_RUN_TEST(test_list_entries);
  310. MU_RUN_TEST(test_entries_deletebyindex);
  311. MU_RUN_TEST(test_entries_delete);
  312. MU_RUN_TEST(test_entry_offset);
  313. }
  314. #define UNUSED(x) (void)x
  315. int main(int argc, char *argv[]) {
  316. UNUSED(argc);
  317. UNUSED(argv);
  318. MU_RUN_SUITE(test_entry_suite);
  319. MU_REPORT();
  320. return MU_EXIT_CODE;
  321. }