11 #include <talloc/talloc.h>
12 #include <types/types.h>
15 #include "device-handler.h"
21 #include "parser-conf.h"
23 #include "parser-test.h"
26 struct list_item list;
27 struct parser *parser;
31 struct discover_device *dev;
39 struct list_item list;
44 void __register_parser(struct parser *parser)
46 struct p_item* i = talloc(NULL, struct p_item);
49 list_add(&parsers, &i->list);
52 static void __attribute__((destructor)) __cleanup_parsers(void)
54 struct p_item *item, *tmp;
56 list_for_each_entry_safe(&parsers, item, tmp, list)
60 static struct discover_device *test_create_device_simple(
61 struct parser_test *test)
66 sprintf(name, "__test%d", dev_idx++);
68 return test_create_device(test, name);
71 struct discover_device *test_create_device(struct parser_test *test,
74 struct discover_device *dev;
76 dev = discover_device_create(test->handler, NULL, name);
78 dev->device->id = talloc_strdup(dev, name);
79 dev->device_path = talloc_asprintf(dev, "/dev/%s", name);
80 dev->mount_path = talloc_asprintf(dev, "/test/mount/%s", name);
86 static struct discover_context *test_create_context(struct parser_test *test)
88 struct discover_context *ctx;
90 ctx = talloc_zero(test, struct discover_context);
93 list_init(&ctx->boot_options);
94 ctx->device = test_create_device_simple(test);
95 ctx->test_data = test;
96 ctx->handler = test->handler;
97 device_handler_add_device(test->handler, ctx->device);
102 /* define our own test platform */
103 static bool test_platform_probe(struct platform *p __attribute__((unused)),
104 void *ctx __attribute__((unused)))
109 struct platform test_platform = {
111 .probe = test_platform_probe,
114 register_platform(test_platform);
116 struct parser_test *test_init(void)
118 struct parser_test *test;
120 test = talloc_zero(NULL, struct parser_test);
122 test->handler = device_handler_init(NULL, NULL, 0);
123 test->ctx = test_create_context(test);
124 list_init(&test->files);
129 void test_fini(struct parser_test *test)
131 device_handler_destroy(test->handler);
136 void __test_read_conf_data(struct parser_test *test,
137 struct discover_device *dev, const char *conf_file,
138 const char *buf, size_t len)
140 test_add_file_data(test, dev, conf_file, buf, len);
143 void test_read_conf_file(struct parser_test *test, const char *filename,
144 const char *conf_file)
152 path = talloc_asprintf(test, "%s/%s", TEST_CONF_BASE, filename);
154 fd = open(path, O_RDONLY);
156 err(EXIT_FAILURE, "Can't open test conf file %s\n", path);
158 rc = fstat(fd, &stat);
163 buf = talloc_array(test, char, size + 1);
165 rc = read(fd, buf, size);
166 assert(rc == (ssize_t)size);
168 *(buf + size) = '\0';
173 test_add_file_data(test, test->ctx->device, conf_file, buf, size);
176 void test_add_file_data(struct parser_test *test, struct discover_device *dev,
177 const char *filename, const void *data, int size)
179 struct test_file *file;
181 file = talloc_zero(test, struct test_file);
182 file->type = TEST_FILE;
184 file->name = filename;
185 file->data = talloc_memdup(test, data, size);
187 list_add(&test->files, &file->list);
190 void test_add_dir(struct parser_test *test, struct discover_device *dev,
193 struct test_file *file;
195 file = talloc_zero(test, struct test_file);
196 file->type = TEST_DIR;
198 file->name = dirname;
199 /* Pick a non-zero size for directories so that "[ -s <dir
200 * path> ]" sees that the file has non-zero size. */
202 list_add(&test->files, &file->list);
205 void test_set_event_source(struct parser_test *test)
207 test->ctx->event = talloc_zero(test->ctx, struct event);
210 void test_set_event_param(struct event *event, const char *name,
213 event_set_param(event, name, value);
216 void test_set_event_device(struct event *event, const char *dev)
218 event->device = talloc_strdup(event, dev);
221 int parser_request_file(struct discover_context *ctx,
222 struct discover_device *dev, const char *filename,
223 char **buf, int *len)
225 struct parser_test *test = ctx->test_data;
226 struct test_file *file;
229 list_for_each_entry(&test->files, file, list) {
230 if (file->dev != dev)
232 if (strcmp(file->name, filename))
234 if (file->type != TEST_FILE)
237 /* the read_file() interface always adds a trailing null
238 * for string-safety; do the same here */
239 tmp = talloc_array(test, char, file->size + 1);
240 memcpy(tmp, file->data, file->size);
241 tmp[file->size] = '\0';
250 int parser_stat_path(struct discover_context *ctx,
251 struct discover_device *dev, const char *path,
252 struct stat *statbuf)
254 struct parser_test *test = ctx->test_data;
255 struct test_file *file;
257 list_for_each_entry(&test->files, file, list) {
258 if (file->dev != dev)
260 if (strcmp(file->name, path))
263 statbuf->st_size = (off_t)file->size;
264 switch (file->type) {
266 statbuf->st_mode = S_IFREG;
269 statbuf->st_mode = S_IFDIR;
272 fprintf(stderr, "%s: bad test file mode %d!", __func__,
283 int parser_replace_file(struct discover_context *ctx,
284 struct discover_device *dev, const char *filename,
287 struct parser_test *test = ctx->test_data;
288 struct test_file *f, *file = NULL;
290 list_for_each_entry(&test->files, f, list) {
293 if (strcmp(f->name, filename))
301 file = talloc_zero(test, struct test_file);
303 file->name = filename;
304 list_add(&test->files, &file->list);
307 file->data = talloc_memdup(test, buf, len);
312 int parser_scandir(struct discover_context *ctx, const char *dirname,
313 struct dirent ***files, int (*filter)(const struct dirent *)
314 __attribute__((unused)),
315 int (*comp)(const struct dirent **, const struct dirent **)
316 __attribute__((unused)))
318 struct parser_test *test = ctx->test_data;
321 struct dirent **dirents = NULL, **new_dirents;
324 list_for_each_entry(&test->files, f, list) {
325 if (f->dev != ctx->device)
328 if (strlen(f->name) <= strlen(dirname))
331 filename = strrchr(f->name, '/');
335 namelen = strlen(filename);
337 if (strncmp(f->name, dirname, strlen(f->name) - namelen))
341 dirents = malloc(sizeof(struct dirent *));
343 new_dirents = realloc(dirents, sizeof(struct dirent *)
348 dirents = new_dirents;
351 dirents[n] = malloc(sizeof(struct dirent) + namelen + 1);
356 strcpy(dirents[n]->d_name, filename + 1);
374 bool parser_is_unique(struct discover_context *ctx, struct discover_device *dev,
375 const char *filename, struct list *found_list)
382 /* Just let the parser process everything. */
386 struct load_url_result *load_url_async(void *ctx, struct pb_url *url,
387 load_url_complete async_cb, void *async_data,
388 waiter_cb stdout_cb, void *stdout_data)
390 struct conf_context *conf = async_data;
391 struct parser_test *test = conf->dc->test_data;
392 struct load_url_result *result;
393 char tmp[] = "/tmp/pb-XXXXXX";
394 ssize_t rc = -1, sz = 0;
395 struct test_file *file;
398 /* Ignore the stdout callback for tests */
407 /* Some parsers will expect to need to read a file, so write the
408 * specified file to a temporary file */
409 list_for_each_entry(&test->files, file, list) {
413 if (strcmp(file->name, url->full))
416 while (sz < file->size) {
417 rc = write(fd, file->data, file->size);
420 "Failed to write to tmpfile, %m\n");
430 result = talloc_zero(ctx, struct load_url_result);
434 result->local = talloc_strdup(result, tmp);
437 result->status = LOAD_ERROR;
439 result->status = result->local ? LOAD_OK : LOAD_ERROR;
440 result->cleanup_local = true;
442 async_cb(result, conf);
447 int parser_request_url(struct discover_context *ctx, struct pb_url *url,
448 char **buf, int *len)
450 struct parser_test *test = ctx->test_data;
451 struct test_file *file;
454 list_for_each_entry(&test->files, file, list) {
458 if (strcmp(file->name, url->full))
461 /* the read_file() interface always adds a trailing null
462 * for string-safety; do the same here */
463 tmp = talloc_array(test, char, file->size + 1);
464 memcpy(tmp, file->data, file->size);
465 tmp[file->size] = '\0';
474 int test_run_parser(struct parser_test *test, const char *parser_name)
478 list_for_each_entry(&parsers, i, list) {
479 if (strcmp(i->parser->name, parser_name))
481 test->ctx->parser = i->parser;
482 return i->parser->parse(test->ctx);
485 errx(EXIT_FAILURE, "%s: parser '%s' not found", __func__, parser_name);
488 bool resource_resolve(struct device_handler *handler, struct parser *parser,
489 struct resource *resource)
493 if (resource->resolved)
497 assert(parser->resolve_resource);
499 return parser->resolve_resource(handler, resource);
502 void boot_option_resolve(struct device_handler *handler,
503 struct discover_boot_option *opt)
505 resource_resolve(handler, opt->source, opt->boot_image);
506 resource_resolve(handler, opt->source, opt->initrd);
507 resource_resolve(handler, opt->source, opt->icon);
510 void test_hotplug_device(struct parser_test *test, struct discover_device *dev)
512 struct discover_boot_option *opt;
514 device_handler_add_device(test->handler, dev);
516 list_for_each_entry(&test->ctx->boot_options, opt, list)
517 boot_option_resolve(test->handler, opt);
520 void test_remove_device(struct parser_test *test, struct discover_device *dev)
522 struct discover_boot_option *opt, *tmp;
524 if (dev == test->ctx->device) {
525 list_for_each_entry_safe(&test->ctx->boot_options,
527 list_remove(&opt->list);
532 device_handler_remove(test->handler, dev);
535 struct discover_boot_option *get_boot_option(struct discover_context *ctx,
538 struct discover_boot_option *opt;
541 list_for_each_entry(&ctx->boot_options, opt, list) {
551 void __check_boot_option_count(struct discover_context *ctx, int count,
552 const char *file, int line)
554 struct discover_boot_option *opt;
555 int defaults = 0, i = 0;
557 list_for_each_entry(&ctx->boot_options, opt, list) {
559 if (opt->option->is_default)
564 fprintf(stderr, "%s:%d: parser returned multiple default "
565 "options\n", file, line);
572 fprintf(stderr, "%s:%d: boot option count check failed\n", file, line);
573 fprintf(stderr, "expected %d options, got %d:\n", count, i);
576 list_for_each_entry(&ctx->boot_options, opt, list)
577 fprintf(stderr, " %2d: %s [%s]\n", i++, opt->option->name,
583 void __check_args(struct discover_boot_option *opt, const char *args,
584 const char *file, int line)
588 if (!opt->option->boot_args && !args)
591 if (!opt->option->boot_args) {
592 fprintf(stderr, "%s:%d: arg check failed\n", file, line);
593 fprintf(stderr, " no arguments parsed\n");
594 fprintf(stderr, " expected '%s'\n", args);
598 rc = strcmp(opt->option->boot_args, args);
600 fprintf(stderr, "%s:%d: arg check failed\n", file, line);
601 fprintf(stderr, " got '%s'\n", opt->option->boot_args);
602 fprintf(stderr, " expected '%s'\n", args);
607 void __check_name(struct discover_boot_option *opt, const char *name,
608 const char *file, int line)
612 rc = strcmp(opt->option->name, name);
614 fprintf(stderr, "%s:%d: name check failed\n", file, line);
615 fprintf(stderr, " got '%s'\n", opt->option->name);
616 fprintf(stderr, " expected '%s'\n", name);
621 void __check_is_default(struct discover_boot_option *opt,
622 const char *file, int line)
624 if (opt->option->is_default)
627 fprintf(stderr, "%s:%d: default check failed\n", file, line);
631 void __check_resolved_local_resource(struct resource *res,
632 struct discover_device *dev, const char *local_path,
633 const char *file, int line)
635 const char *exp_url, *got_url;
638 errx(EXIT_FAILURE, "%s:%d: No resource", file, line);
641 errx(EXIT_FAILURE, "%s:%d: Resource is not resolved",
644 exp_url = talloc_asprintf(res, "file://%s%s",
645 dev->mount_path, local_path);
646 got_url = pb_url_to_string(res->url);
648 if (strcmp(got_url, exp_url)) {
649 fprintf(stderr, "%s:%d: Resource mismatch\n", file, line);
650 fprintf(stderr, " got '%s'\n", got_url);
651 fprintf(stderr, " expected '%s'\n", exp_url);
656 void __check_resolved_url_resource(struct resource *res,
657 const char *url, const char *file, int line)
662 errx(EXIT_FAILURE, "%s:%d: No resource", file, line);
665 errx(EXIT_FAILURE, "%s:%d: Resource is not resolved",
668 res_url = pb_url_to_string(res->url);
669 if (strcmp(url, res_url)) {
670 fprintf(stderr, "%s:%d: Resource mismatch\n", file, line);
671 fprintf(stderr, " got '%s'\n", res_url);
672 fprintf(stderr, " expected '%s'\n", url);
676 void __check_unresolved_resource(struct resource *res,
677 const char *file, int line)
680 errx(EXIT_FAILURE, "%s:%d: No resource", file, line);
683 errx(EXIT_FAILURE, "%s:%d: Resource is resolved", file, line);
686 void __check_not_present_resource(struct resource *res,
687 const char *file, int line)
690 errx(EXIT_FAILURE, "%s:%d: Resource present", file, line);
693 static void dump_file_data(const void *buf, int len)
695 int i, j, hex_len = strlen("00 ");
696 const int row_len = 16;
698 for (i = 0; i < len; i += row_len) {
699 char hbuf[row_len * hex_len + 1];
700 char cbuf[row_len + strlen("|") + 1];
702 for (j = 0; (j < row_len) && ((i+j) < len); j++) {
703 char c = ((const char *)buf)[i + j];
705 snprintf(hbuf + j * hex_len, hex_len + 1, "%02x ", c);
710 snprintf(cbuf + j, hex_len + 1, "%c", c);
715 fprintf(stderr, "%08x %*s |%s\n", i,
716 0 - (int)sizeof(hbuf) + 1, hbuf, cbuf);
720 void __check_file_contents(struct parser_test *test,
721 struct discover_device *dev, const char *filename,
722 const char *buf, int len,
723 const char *srcfile, int srcline)
725 struct test_file *f, *file = NULL;
727 list_for_each_entry(&test->files, f, list) {
730 if (strcmp(f->name, filename))
738 errx(EXIT_FAILURE, "%s:%d: File '%s' not found",
739 srcfile, srcline, filename);
741 if (file->size != len || memcmp(file->data, buf, len)) {
742 fprintf(stderr, "%s:%d: File '%s' data/size mismatch\n",
743 srcfile, srcline, filename);
744 fprintf(stderr, "Expected:\n");
745 dump_file_data(buf, len);
746 fprintf(stderr, "Got:\n");
747 dump_file_data(file->data, file->size);