#define OWL_PERL #define WINDOW FAKE_WINDOW #include "owl.h" #undef WINDOW #include #include #undef instr #include owl_global g; int numtests; int owl_regtest(void); int owl_util_regtest(void); int owl_dict_regtest(void); int owl_variable_regtest(void); int owl_filter_regtest(void); int owl_obarray_regtest(void); int owl_editwin_regtest(void); int owl_fmtext_regtest(void); int owl_smartfilter_regtest(void); extern void owl_perl_xs_init(pTHX); int main(int argc, char **argv, char **env) { FILE *rnull; FILE *wnull; char *perlerr; int status = 0; SCREEN *screen; if (argc <= 1) { fprintf(stderr, "Usage: %s --builtin|TEST.t|-le CODE\n", argv[0]); return 1; } /* initialize a fake ncurses, detached from std{in,out} */ wnull = fopen("/dev/null", "w"); rnull = fopen("/dev/null", "r"); screen = newterm("xterm", wnull, rnull); /* initialize global structures */ owl_global_init(&g); perlerr = owl_perlconfig_initperl(NULL, &argc, &argv, &env); if (perlerr) { endwin(); fprintf(stderr, "Internal perl error: %s\n", perlerr); status = 1; goto out; } owl_global_complete_setup(&g); owl_global_setup_default_filters(&g); owl_view_create(owl_global_get_current_view(&g), "main", owl_global_get_filter(&g, "all"), owl_global_get_style_by_name(&g, "default")); owl_function_firstmsg(); ENTER; SAVETMPS; if (strcmp(argv[1], "--builtin") == 0) { status = owl_regtest(); } else if (strcmp(argv[1], "-le") == 0 && argc > 2) { /* * 'prove' runs its harness perl with '-le CODE' to get some * information out. */ moreswitches("l"); eval_pv(argv[2], true); } else { sv_setpv(get_sv("0", false), argv[1]); sv_setpv(get_sv("main::test_prog", TRUE), argv[1]); eval_pv("do $main::test_prog; die($@) if($@)", true); } status = 0; FREETMPS; LEAVE; out: perl_destruct(owl_global_get_perlinterp(&g)); perl_free(owl_global_get_perlinterp(&g)); /* probably not necessary, but tear down the screen */ endwin(); delscreen(screen); fclose(rnull); fclose(wnull); return status; } int owl_regtest(void) { numtests = 0; int numfailures=0; /* printf("1..%d\n", OWL_UTIL_NTESTS+OWL_DICT_NTESTS+OWL_VARIABLE_NTESTS +OWL_FILTER_NTESTS+OWL_OBARRAY_NTESTS); */ numfailures += owl_util_regtest(); numfailures += owl_dict_regtest(); numfailures += owl_variable_regtest(); numfailures += owl_filter_regtest(); numfailures += owl_editwin_regtest(); numfailures += owl_fmtext_regtest(); numfailures += owl_smartfilter_regtest(); if (numfailures) { fprintf(stderr, "# *** WARNING: %d failures total\n", numfailures); } printf("1..%d\n", numtests); return(numfailures); } #define FAIL_UNLESS(desc,pred) do { int __pred = (pred); \ numtests++; \ printf("%s %s", (__pred)?"ok":(numfailed++,"not ok"), desc); \ if(!(__pred)) printf("\t(%s:%d)", __FILE__, __LINE__); printf("%c", '\n'); } while(0) int owl_util_regtest(void) { int numfailed=0; printf("# BEGIN testing owl_util\n"); #define CHECK_STR_AND_FREE(desc, expected, expr) \ do { \ char *__value = (expr); \ FAIL_UNLESS((desc), !strcmp((expected), __value)); \ g_free(__value); \ } while (0) CHECK_STR_AND_FREE("owl_text_substitute 2", "fYZYZ", owl_text_substitute("foo", "o", "YZ")); CHECK_STR_AND_FREE("owl_text_substitute 3", "foo", owl_text_substitute("fYZYZ", "YZ", "o")); CHECK_STR_AND_FREE("owl_text_substitute 4", "/u/foo/meep", owl_text_substitute("~/meep", "~", "/u/foo")); FAIL_UNLESS("skiptokens 1", !strcmp("bar quux", skiptokens("foo bar quux", 1))); FAIL_UNLESS("skiptokens 2", !strcmp("meep", skiptokens("foo 'bar quux' meep", 2))); CHECK_STR_AND_FREE("expand_tabs 1", " hi", owl_text_expand_tabs("\thi")); CHECK_STR_AND_FREE("expand_tabs 2", " hi\nword tab", owl_text_expand_tabs("\thi\nword\ttab")); CHECK_STR_AND_FREE("expand_tabs 3", " 2 tabs", owl_text_expand_tabs("\t\t2 tabs")); CHECK_STR_AND_FREE("expand_tabs 4", "α ααααααα! ", owl_text_expand_tabs("α\tααααααα!\t")); CHECK_STR_AND_FREE("expand_tabs 5", "A AAA!! ", owl_text_expand_tabs("A\tAAA!!\t")); FAIL_UNLESS("skiptokens 1", !strcmp("world", skiptokens("hello world", 1))); FAIL_UNLESS("skiptokens 2", !strcmp("c d e", skiptokens("a b c d e", 2))); FAIL_UNLESS("skiptokens 3", !strcmp("\"b\" c d e", skiptokens("a \"b\" c d e", 1))); FAIL_UNLESS("skiptokens 4", !strcmp("c d e", skiptokens("a \"b\" c d e", 2))); FAIL_UNLESS("skiptokens 5", !strcmp("c d e", skiptokens("a \"'\" c d e", 2))); #define CHECK_QUOTING(desc, unquoted, quoted) \ do { \ int __argc; \ char *__quoted = owl_arg_quote(unquoted); \ char **__argv; \ FAIL_UNLESS(desc, !strcmp(quoted, __quoted)); \ __argv = owl_parseline(__quoted, &__argc); \ FAIL_UNLESS(desc " - arg count", __argc == 1); \ FAIL_UNLESS(desc " - null-terminated", \ __argv[__argc] == NULL); \ FAIL_UNLESS(desc " - parsed", \ !strcmp(__argv[0], unquoted)); \ g_strfreev(__argv); \ g_free(__quoted); \ } while (0) CHECK_QUOTING("boring text", "mango", "mango"); CHECK_QUOTING("spaces", "mangos are tasty", "'mangos are tasty'"); CHECK_QUOTING("single quotes", "mango's", "\"mango's\""); CHECK_QUOTING("double quotes", "he said \"mangos are tasty\"", "'he said \"mangos are tasty\"'"); CHECK_QUOTING("both quotes", "he said \"mango's are tasty even when you put in " "a random apostrophe\"", "\"he said \"'\"'\"mango's are tasty even when you put in " "a random apostrophe\"'\"'\"\""); CHECK_QUOTING("quote monster", "'\"\"'\"'''\"", "\"" "'" "\"'\"'\"" "\"'\"'\"" "'" "\"'\"'\"" "'" "'" "'" "\"'\"'\"" "\""); GString *g = g_string_new(""); owl_string_appendf_quoted(g, "%q foo %q%q %s %", "hello", "world is", "can't"); FAIL_UNLESS("owl_string_appendf", !strcmp(g->str, "hello foo 'world is'\"can't\" %s %")); g_string_free(g, true); /* if (numfailed) printf("*** WARNING: failures encountered with owl_util\n"); */ printf("# END testing owl_util (%d failures)\n", numfailed); return(numfailed); } int owl_dict_regtest(void) { owl_dict d; owl_list l; int numfailed=0; char *av="aval", *bv="bval", *cv="cval", *dv="dval"; printf("# BEGIN testing owl_dict\n"); owl_dict_create(&d); FAIL_UNLESS("insert b", 0==owl_dict_insert_element(&d, "b", bv, owl_dict_noop_delete)); FAIL_UNLESS("insert d", 0==owl_dict_insert_element(&d, "d", dv, owl_dict_noop_delete)); FAIL_UNLESS("insert a", 0==owl_dict_insert_element(&d, "a", av, owl_dict_noop_delete)); FAIL_UNLESS("insert c", 0==owl_dict_insert_element(&d, "c", cv, owl_dict_noop_delete)); FAIL_UNLESS("reinsert d (no replace)", -2==owl_dict_insert_element(&d, "d", dv, 0)); FAIL_UNLESS("find a", av==owl_dict_find_element(&d, "a")); FAIL_UNLESS("find b", bv==owl_dict_find_element(&d, "b")); FAIL_UNLESS("find c", cv==owl_dict_find_element(&d, "c")); FAIL_UNLESS("find d", dv==owl_dict_find_element(&d, "d")); FAIL_UNLESS("find e (non-existent)", NULL==owl_dict_find_element(&d, "e")); FAIL_UNLESS("remove d", dv==owl_dict_remove_element(&d, "d")); FAIL_UNLESS("find d (post-removal)", NULL==owl_dict_find_element(&d, "d")); FAIL_UNLESS("get_size", 3==owl_dict_get_size(&d)); owl_list_create(&l); owl_dict_get_keys(&d, &l); FAIL_UNLESS("get_keys result size", 3==owl_list_get_size(&l)); /* these assume the returned keys are sorted */ FAIL_UNLESS("get_keys result val",0==strcmp("a",owl_list_get_element(&l,0))); FAIL_UNLESS("get_keys result val",0==strcmp("b",owl_list_get_element(&l,1))); FAIL_UNLESS("get_keys result val",0==strcmp("c",owl_list_get_element(&l,2))); owl_list_cleanup(&l, g_free); owl_dict_cleanup(&d, NULL); /* if (numfailed) printf("*** WARNING: failures encountered with owl_dict\n"); */ printf("# END testing owl_dict (%d failures)\n", numfailed); return(numfailed); } int owl_variable_regtest(void) { owl_vardict vd; int numfailed=0; char *value; const void *v; printf("# BEGIN testing owl_variable\n"); FAIL_UNLESS("setup", 0==owl_variable_dict_setup(&vd)); FAIL_UNLESS("get bool", 0==owl_variable_get_bool(&vd,"rxping")); FAIL_UNLESS("get bool (no such)", -1==owl_variable_get_bool(&vd,"mumble")); FAIL_UNLESS("get bool as string", !strcmp((value = owl_variable_get_tostring(&vd,"rxping")), "off")); g_free(value); FAIL_UNLESS("set bool 1", 0==owl_variable_set_bool_on(&vd,"rxping")); FAIL_UNLESS("get bool 2", 1==owl_variable_get_bool(&vd,"rxping")); FAIL_UNLESS("set bool 3", 0==owl_variable_set_fromstring(&vd,"rxping","off",0,0)); FAIL_UNLESS("get bool 4", 0==owl_variable_get_bool(&vd,"rxping")); FAIL_UNLESS("set bool 5", -1==owl_variable_set_fromstring(&vd,"rxping","xxx",0,0)); FAIL_UNLESS("get bool 6", 0==owl_variable_get_bool(&vd,"rxping")); FAIL_UNLESS("get string", 0==strcmp("~/zlog/people", owl_variable_get_string(&vd,"logpath"))); FAIL_UNLESS("set string 7", 0==owl_variable_set_string(&vd,"logpath","whee")); FAIL_UNLESS("get string", 0==strcmp("whee", owl_variable_get_string(&vd,"logpath"))); FAIL_UNLESS("get int", 8==owl_variable_get_int(&vd,"typewinsize")); FAIL_UNLESS("get int (no such)", -1==owl_variable_get_int(&vd,"mmble")); FAIL_UNLESS("get int as string", !strcmp((value = owl_variable_get_tostring(&vd,"typewinsize")), "8")); g_free(value); FAIL_UNLESS("set int 1", 0==owl_variable_set_int(&vd,"typewinsize",12)); FAIL_UNLESS("get int 2", 12==owl_variable_get_int(&vd,"typewinsize")); FAIL_UNLESS("set int 1b", -1==owl_variable_set_int(&vd,"typewinsize",-3)); FAIL_UNLESS("get int 2b", 12==owl_variable_get_int(&vd,"typewinsize")); FAIL_UNLESS("set int 3", 0==owl_variable_set_fromstring(&vd,"typewinsize","9",0,0)); FAIL_UNLESS("get int 4", 9==owl_variable_get_int(&vd,"typewinsize")); FAIL_UNLESS("set int 5", -1==owl_variable_set_fromstring(&vd,"typewinsize","xxx",0,0)); FAIL_UNLESS("set int 6", -1==owl_variable_set_fromstring(&vd,"typewinsize","",0,0)); FAIL_UNLESS("get int 7", 9==owl_variable_get_int(&vd,"typewinsize")); owl_variable_dict_newvar_string(&vd, "stringvar", "", "", "testval"); FAIL_UNLESS("get new string var", NULL != (v = owl_variable_get(&vd, "stringvar", OWL_VARIABLE_STRING))); FAIL_UNLESS("get new string val", !strcmp("testval", owl_variable_get_string(&vd, "stringvar"))); owl_variable_set_string(&vd, "stringvar", "new val"); FAIL_UNLESS("update string val", !strcmp("new val", owl_variable_get_string(&vd, "stringvar"))); owl_variable_dict_newvar_int(&vd, "intvar", "", "", 47); FAIL_UNLESS("get new int var", NULL != (v = owl_variable_get(&vd, "intvar", OWL_VARIABLE_INT))); FAIL_UNLESS("get new int val", 47 == owl_variable_get_int(&vd, "intvar")); owl_variable_set_int(&vd, "intvar", 17); FAIL_UNLESS("update bool val", 17 == owl_variable_get_int(&vd, "intvar")); owl_variable_dict_newvar_bool(&vd, "boolvar", "", "", 1); FAIL_UNLESS("get new bool var", NULL != (v = owl_variable_get(&vd, "boolvar", OWL_VARIABLE_BOOL))); FAIL_UNLESS("get new bool val", owl_variable_get_bool(&vd, "boolvar")); owl_variable_set_bool_off(&vd, "boolvar"); FAIL_UNLESS("update string val", !owl_variable_get_bool(&vd, "boolvar")); owl_variable_dict_cleanup(&vd); /* if (numfailed) printf("*** WARNING: failures encountered with owl_variable\n"); */ printf("# END testing owl_variable (%d failures)\n", numfailed); return(numfailed); } static int owl_filter_test_string(const char *filt, const owl_message *m, int shouldmatch) { owl_filter *f; int ok; int failed = 0; if ((f = owl_filter_new_fromstring("test-filter", filt)) == NULL) { printf("not ok can't parse %s\n", filt); failed = 1; goto out; } ok = owl_filter_message_match(f, m); if((shouldmatch && !ok) || (!shouldmatch && ok)) { printf("not ok match %s (got %d, expected %d)\n", filt, ok, shouldmatch); failed = 1; } out: owl_filter_delete(f); if(!failed) { printf("ok %s %s\n", shouldmatch ? "matches" : "doesn't match", filt); } return failed; } int owl_filter_regtest(void) { int numfailed=0; owl_message m; owl_filter *f1, *f2, *f3, *f4, *f5; owl_message_init(&m); owl_message_set_type_zephyr(&m); owl_message_set_direction_in(&m); owl_message_set_class(&m, "owl"); owl_message_set_instance(&m, "tester"); owl_message_set_sender(&m, "owl-user"); owl_message_set_recipient(&m, "joe"); owl_message_set_attribute(&m, "foo", "bar"); #define TEST_FILTER(f, e) do { \ numtests++; \ numfailed += owl_filter_test_string(f, &m, e); \ } while(0) TEST_FILTER("true", 1); TEST_FILTER("false", 0); TEST_FILTER("( true )", 1); TEST_FILTER("not false", 1); TEST_FILTER("( true ) or ( false )", 1); TEST_FILTER("true and false", 0); TEST_FILTER("( true or true ) or ( ( false ) )", 1); TEST_FILTER("class owl", 1); TEST_FILTER("class ^owl$", 1); TEST_FILTER("instance test", 1); TEST_FILTER("instance ^test$", 0); TEST_FILTER("instance ^tester$", 1); TEST_FILTER("foo bar", 1); TEST_FILTER("class owl and instance tester", 1); TEST_FILTER("type ^zephyr$ and direction ^in$ and ( class ^owl$ or instance ^owl$ )", 1); /* Order of operations and precedence */ TEST_FILTER("not true or false", 0); TEST_FILTER("true or true and false", 0); TEST_FILTER("true and true and false or true", 1); TEST_FILTER("false and false or true", 1); TEST_FILTER("true and false or false", 0); f1 = owl_filter_new_fromstring("f1", "class owl"); owl_global_add_filter(&g, f1); TEST_FILTER("filter f1", 1); owl_global_remove_filter(&g, "f1"); /* Test recursion prevention */ FAIL_UNLESS("self reference", (f2 = owl_filter_new_fromstring("test", "filter test")) == NULL); owl_filter_delete(f2); /* mutual recursion */ f3 = owl_filter_new_fromstring("f3", "filter f4"); owl_global_add_filter(&g, f3); FAIL_UNLESS("mutual recursion", (f4 = owl_filter_new_fromstring("f4", "filter f3")) == NULL); owl_global_remove_filter(&g, "f3"); owl_filter_delete(f4); /* support referencing a filter several times */ FAIL_UNLESS("DAG", (f5 = owl_filter_new_fromstring("dag", "filter f1 or filter f1")) != NULL); owl_filter_delete(f5); return 0; } int owl_editwin_regtest(void) { int numfailed = 0; const char *p; owl_editwin *oe; const char *autowrap_string = "we feel our owls should live " "closer to our ponies."; printf("# BEGIN testing owl_editwin\n"); oe = owl_editwin_new(NULL, 80, 80, OWL_EDITWIN_STYLE_MULTILINE, NULL); /* TODO: make the strings a little more lenient w.r.t trailing whitespace */ /* check paragraph fill */ owl_editwin_insert_string(oe, "blah blah blah blah blah blah blah blah blah blah blah blah blah blah blah blah blah.\n\nblah"); owl_editwin_move_to_top(oe); owl_editwin_fill_paragraph(oe); p = owl_editwin_get_text(oe); FAIL_UNLESS("text was correctly wrapped", p && !strcmp(p, "blah blah blah blah blah blah blah blah blah blah blah blah blah blah\n" "blah blah blah.\n" "\n" "blah")); owl_editwin_unref(oe); oe = NULL; oe = owl_editwin_new(NULL, 80, 80, OWL_EDITWIN_STYLE_MULTILINE, NULL); /* check that lines ending with ". " correctly fill */ owl_editwin_insert_string(oe, "blah blah blah blah blah blah blah blah blah blah blah blah blah blah blah blah blah. \n\nblah"); owl_editwin_move_to_top(oe); owl_editwin_fill_paragraph(oe); p = owl_editwin_get_text(oe); FAIL_UNLESS("text was correctly wrapped", p && !strcmp(p, "blah blah blah blah blah blah blah blah blah blah blah blah blah blah\n" "blah blah blah. \n" "\n" "blah")); owl_editwin_unref(oe); oe = NULL; /* Test owl_editwin_move_to_beginning_of_line. */ oe = owl_editwin_new(NULL, 80, 80, OWL_EDITWIN_STYLE_MULTILINE, NULL); owl_editwin_insert_string(oe, "\n"); owl_editwin_insert_string(oe, "12345678\n"); owl_editwin_insert_string(oe, "\n"); owl_editwin_insert_string(oe, "abcdefg\n"); owl_editwin_move_to_top(oe); FAIL_UNLESS("already at beginning of line", owl_editwin_move_to_beginning_of_line(oe) == 0); owl_editwin_line_move(oe, 1); owl_editwin_point_move(oe, 5); FAIL_UNLESS("find beginning of line after empty first line", owl_editwin_move_to_beginning_of_line(oe) == -5); owl_editwin_line_move(oe, 1); FAIL_UNLESS("find beginning empty middle line", owl_editwin_move_to_beginning_of_line(oe) == 0); owl_editwin_line_move(oe, 1); owl_editwin_point_move(oe, 2); FAIL_UNLESS("find beginning of line after empty middle line", owl_editwin_move_to_beginning_of_line(oe) == -2); owl_editwin_unref(oe); oe = NULL; /* Test automatic line-wrapping. */ owl_global_set_edit_maxwrapcols(&g, 10); oe = owl_editwin_new(NULL, 80, 80, OWL_EDITWIN_STYLE_MULTILINE, NULL); for (p = autowrap_string; *p; p++) { owl_input j; j.ch = *p; j.uch = *p; /* Assuming ASCII. */ owl_editwin_process_char(oe, j); } p = owl_editwin_get_text(oe); FAIL_UNLESS("text was automatically wrapped", p && !strcmp(p, "we feel\n" "our owls\n" "should\n" "live\n" "closer to\n" "our\n" "ponies.")); owl_editwin_unref(oe); oe = NULL; owl_global_set_edit_maxwrapcols(&g, 70); /* Test owl_editwin_current_column. */ oe = owl_editwin_new(NULL, 80, 80, OWL_EDITWIN_STYLE_MULTILINE, NULL); FAIL_UNLESS("initial column zero", owl_editwin_current_column(oe) == 0); owl_editwin_insert_string(oe, "abcdef"); FAIL_UNLESS("simple insert", owl_editwin_current_column(oe) == 6); owl_editwin_insert_string(oe, "\t"); FAIL_UNLESS("insert tabs", owl_editwin_current_column(oe) == 8); owl_editwin_insert_string(oe, "123\n12\t3"); FAIL_UNLESS("newline with junk", owl_editwin_current_column(oe) == 9); owl_editwin_move_to_beginning_of_line(oe); FAIL_UNLESS("beginning of line", owl_editwin_current_column(oe) == 0); owl_editwin_unref(oe); oe = NULL; printf("# END testing owl_editwin (%d failures)\n", numfailed); return numfailed; } int owl_fmtext_regtest(void) { int numfailed = 0; int start, end; owl_fmtext fm1; owl_fmtext fm2; owl_regex re; char *str; printf("# BEGIN testing owl_fmtext\n"); owl_fmtext_init_null(&fm1); owl_fmtext_init_null(&fm2); /* Verify text gets correctly appended. */ owl_fmtext_append_normal(&fm1, "1234567898"); owl_fmtext_append_fmtext(&fm2, &fm1); FAIL_UNLESS("string lengths correct", owl_fmtext_num_bytes(&fm2) == strlen(owl_fmtext_get_text(&fm2))); /* Test owl_fmtext_num_lines. */ owl_fmtext_clear(&fm1); FAIL_UNLESS("empty line correct", owl_fmtext_num_lines(&fm1) == 0); owl_fmtext_append_normal(&fm1, "12345\n67898"); FAIL_UNLESS("trailing chars correct", owl_fmtext_num_lines(&fm1) == 2); owl_fmtext_append_normal(&fm1, "\n"); FAIL_UNLESS("trailing newline correct", owl_fmtext_num_lines(&fm1) == 2); owl_fmtext_append_bold(&fm1, ""); FAIL_UNLESS("trailing attributes correct", owl_fmtext_num_lines(&fm1) == 2); /* Test owl_fmtext_truncate_lines */ owl_fmtext_clear(&fm1); owl_fmtext_append_normal(&fm1, "0\n1\n2\n3\n4\n"); owl_fmtext_clear(&fm2); owl_fmtext_truncate_lines(&fm1, 1, 3, &fm2); str = owl_fmtext_print_plain(&fm2); FAIL_UNLESS("lines corrected truncated", str && !strcmp(str, "1\n2\n3\n")); g_free(str); owl_fmtext_clear(&fm2); owl_fmtext_truncate_lines(&fm1, 1, 5, &fm2); str = owl_fmtext_print_plain(&fm2); FAIL_UNLESS("lines corrected truncated", str && !strcmp(str, "1\n2\n3\n4\n")); g_free(str); /* Test owl_fmtext_truncate_cols. */ owl_fmtext_clear(&fm1); owl_fmtext_append_normal(&fm1, "123456789012345\n"); owl_fmtext_append_normal(&fm1, "123456789\n"); owl_fmtext_append_normal(&fm1, "1234567890\n"); owl_fmtext_clear(&fm2); owl_fmtext_truncate_cols(&fm1, 4, 9, &fm2); str = owl_fmtext_print_plain(&fm2); FAIL_UNLESS("columns correctly truncated", str && !strcmp(str, "567890" "56789\n" "567890")); g_free(str); owl_fmtext_clear(&fm1); owl_fmtext_append_normal(&fm1, "12\t1234"); owl_fmtext_append_bold(&fm1, "56\n"); owl_fmtext_append_bold(&fm1, "12345678\t\n"); owl_fmtext_clear(&fm2); owl_fmtext_truncate_cols(&fm1, 4, 13, &fm2); str = owl_fmtext_print_plain(&fm2); FAIL_UNLESS("columns correctly truncated", str && !strcmp(str, " 123456" "5678 ")); g_free(str); /* Test owl_fmtext_expand_tabs. */ owl_fmtext_clear(&fm1); owl_fmtext_append_normal(&fm1, "12\t1234"); owl_fmtext_append_bold(&fm1, "567\t1\n12345678\t1"); owl_fmtext_clear(&fm2); owl_fmtext_expand_tabs(&fm1, &fm2, 0); str = owl_fmtext_print_plain(&fm2); FAIL_UNLESS("no tabs remaining", strchr(str, '\t') == NULL); FAIL_UNLESS("tabs corrected expanded", str && !strcmp(str, "12 1234567 1\n" "12345678 1")); g_free(str); owl_fmtext_clear(&fm2); owl_fmtext_expand_tabs(&fm1, &fm2, 1); str = owl_fmtext_print_plain(&fm2); FAIL_UNLESS("no tabs remaining", strchr(str, '\t') == NULL); FAIL_UNLESS("tabs corrected expanded", str && !strcmp(str, "12 1234567 1\n" "12345678 1")); g_free(str); /* Test owl_fmtext_search. */ owl_fmtext_clear(&fm1); owl_fmtext_append_normal(&fm1, "123123123123"); owl_regex_create(&re, "12"); { int count = 0, offset; offset = owl_fmtext_search(&fm1, &re, 0); while (offset >= 0) { FAIL_UNLESS("search matches", !strncmp("12", owl_fmtext_get_text(&fm1) + offset, 2)); count++; offset = owl_fmtext_search(&fm1, &re, offset+1); } FAIL_UNLESS("exactly four matches", count == 4); } owl_regex_cleanup(&re); /* Test owl_fmtext_line_number. */ owl_fmtext_clear(&fm1); owl_fmtext_append_normal(&fm1, "123\n456\n"); owl_fmtext_append_bold(&fm1, ""); FAIL_UNLESS("lines start at 0", 0 == owl_fmtext_line_number(&fm1, 0)); FAIL_UNLESS("trailing formatting characters part of false line", 2 == owl_fmtext_line_number(&fm1, owl_fmtext_num_bytes(&fm1))); owl_regex_create_quoted(&re, "456"); FAIL_UNLESS("correctly find second line (line 1)", 1 == owl_fmtext_line_number(&fm1, owl_fmtext_search(&fm1, &re, 0))); owl_regex_cleanup(&re); /* Test owl_fmtext_line_extents. */ owl_fmtext_clear(&fm1); owl_fmtext_append_normal(&fm1, "123\n456\n789"); owl_fmtext_line_extents(&fm1, 1, &start, &end); FAIL_UNLESS("line contents", !strncmp("456\n", owl_fmtext_get_text(&fm1)+start, end-start)); owl_fmtext_line_extents(&fm1, 2, &start, &end); FAIL_UNLESS("point to end of buffer", end == owl_fmtext_num_bytes(&fm1)); owl_fmtext_cleanup(&fm1); owl_fmtext_cleanup(&fm2); printf("# END testing owl_fmtext (%d failures)\n", numfailed); return numfailed; } static int owl_smartfilter_test_equals(const char *filtname, const char *expected) { owl_filter *f = NULL; char *filtstr = NULL; int failed = 0; f = owl_global_get_filter(&g, filtname); if (f == NULL) { printf("not ok filter missing: %s\n", filtname); failed = 1; goto out; } /* TODO: Come up with a better way to test this. */ filtstr = owl_filter_print(f); if (strcmp(expected, filtstr)) { printf("not ok filter incorrect: |%s| instead of |%s|\n", filtstr, expected); failed = 1; goto out; } out: g_free(filtstr); return failed; } static int owl_classinstfilt_test(const char *c, const char *i, int related, const char *expected) { char *filtname = NULL; int failed = 0; filtname = owl_function_classinstfilt(c, i, related); if (filtname == NULL) { printf("not ok null filtname: %s %s %s\n", c, i ? i : "(null)", related ? "related" : "not related"); failed = 1; goto out; } if (owl_smartfilter_test_equals(filtname, expected)) { failed = 1; goto out; } out: if (!failed) { printf("ok %s\n", filtname); } if (filtname) owl_global_remove_filter(&g, filtname); g_free(filtname); return failed; } static int owl_zuserfilt_test(const char *longuser, const char *expected) { char *filtname = NULL; int failed = 0; filtname = owl_function_zuserfilt(longuser); if (filtname == NULL) { printf("not ok null filtname: %s\n", longuser); failed = 1; goto out; } if (owl_smartfilter_test_equals(filtname, expected)) { failed = 1; goto out; } out: if (!failed) { printf("ok %s\n", filtname); } if (filtname) owl_global_remove_filter(&g, filtname); g_free(filtname); return failed; } int owl_smartfilter_regtest(void) { int numfailed = 0; printf("# BEGIN testing owl_smartfilter\n"); /* Check classinst making. */ #define TEST_CLASSINSTFILT(c, i, r, e) do { \ numtests++; \ numfailed += owl_classinstfilt_test(c, i, r, e); \ } while (0) TEST_CLASSINSTFILT("message", NULL, false, "class ^message$\n"); TEST_CLASSINSTFILT("message", NULL, true, "class ^(un)*message(\\.d)*$\n"); TEST_CLASSINSTFILT("message", "personal", false, "class ^message$ and instance ^personal$\n"); TEST_CLASSINSTFILT("message", "personal", true, "class ^(un)*message(\\.d)*$ and ( instance ^(un)*personal(\\.d)*$ )\n"); TEST_CLASSINSTFILT("message", "evil\tinstance", false, "class ^message$ and instance '^evil\tinstance$'\n"); TEST_CLASSINSTFILT("message", "evil instance", false, "class ^message$ and instance '^evil instance$'\n"); TEST_CLASSINSTFILT("message", "evil'instance", false, "class ^message$ and instance \"^evil'instance$\"\n"); TEST_CLASSINSTFILT("message", "evil\"instance", false, "class ^message$ and instance '^evil\"instance$'\n"); TEST_CLASSINSTFILT("message", "evil$instance", false, "class ^message$ and instance ^evil\\$instance$\n"); #define TEST_ZUSERFILT(l, e) do { \ numtests++; \ numfailed += owl_zuserfilt_test(l, e); \ } while (0) TEST_ZUSERFILT("user", "( type ^zephyr$ and filter personal and " "( ( direction ^in$ and sender " "^user$" " ) or ( direction ^out$ and recipient " "^user$" " ) ) ) or ( ( class ^login$ ) and ( sender " "^user$" " ) )\n"); TEST_ZUSERFILT("very evil\t.user", "( type ^zephyr$ and filter personal and " "( ( direction ^in$ and sender " "'^very evil\t\\.user$'" " ) or ( direction ^out$ and recipient " "'^very evil\t\\.user$'" " ) ) ) or ( ( class ^login$ ) and ( sender " "'^very evil\t\\.user$'" " ) )\n"); printf("# END testing owl_smartfilter (%d failures)\n", numfailed); return numfailed; }