chiark / gitweb /
Automatically rescan when a filesystem is mounted or unmounted.
[disorder] / libtests / test.h
1 /*
2  * This file is part of DisOrder.
3  * Copyright (C) 2005, 2007, 2008 Richard Kettlewell
4  *
5  * This program is free software: you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation, either version 3 of the License, or
8  * (at your option) any later version.
9  * 
10  * This program is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  * GNU General Public License for more details.
14  * 
15  * You should have received a copy of the GNU General Public License
16  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
17  */
18 /** @file libtests/test.h @brief Library tests */
19
20 #ifndef TEST_H
21 #define TEST_H
22
23 #include "common.h"
24
25 #include <errno.h>
26 #include <ctype.h>
27 #include <sys/types.h>
28 #include <sys/stat.h>
29 #include <unistd.h>
30 #include <signal.h>
31 #include <stddef.h>
32 #include <sys/socket.h>
33 #include <netdb.h>
34 #include <netinet/in.h>
35 #include <sys/un.h>
36 #include <pcre.h>
37 #include <setjmp.h>
38
39 #include "mem.h"
40 #include "log.h"
41 #include "vector.h"
42 #include "charset.h"
43 #include "mime.h"
44 #include "hex.h"
45 #include "heap.h"
46 #include "unicode.h"
47 #include "inputline.h"
48 #include "wstat.h"
49 #include "signame.h"
50 #include "cache.h"
51 #include "filepart.h"
52 #include "hash.h"
53 #include "selection.h"
54 #include "syscalls.h"
55 #include "kvp.h"
56 #include "sink.h"
57 #include "printf.h"
58 #include "basen.h"
59 #include "split.h"
60 #include "configuration.h"
61 #include "addr.h"
62 #include "base64.h"
63 #include "url.h"
64 #include "regsub.h"
65
66 extern long long tests, errors;
67 extern int fail_first;
68 extern int verbose;
69 extern int skipped;
70
71 /** @brief Checks that @p expr is nonzero
72  * @param expr Expression to check
73  *
74  * If @p expr is nonzero then logs an error (and continues).
75  */
76 #define insist(expr) do {                               \
77   if(!(expr)) {                                         \
78     count_error();                                              \
79     fprintf(stderr, "%s:%d: error checking %s\n",       \
80             __FILE__, __LINE__, #expr);                 \
81   }                                                     \
82   ++tests;                                              \
83 } while(0)
84
85 /** @brief Check that a pair of strings match
86  * @param GOT What we actually got
87  * @param WANT What we wanted
88  *
89  * If @p GOT and @p WANT differ then logs an error (and continues).
90  *
91  * @p WANT is allowed to evaluate to a null pointer (but if it comes to
92  * anything else then must be safe to strcmp).
93  */
94 #define check_string(GOT, WANT) do {                                    \
95   const char *got = GOT;                                                \
96   const char *want = WANT;                                              \
97                                                                         \
98   if(want == 0) {                                                       \
99     fprintf(stderr, "%s:%d: %s returned 0\n",                           \
100             __FILE__, __LINE__, #GOT);                                  \
101     count_error();                                                      \
102   } else if(strcmp(want, got)) {                                        \
103     fprintf(stderr, "%s:%d: %s returned:\n%s\nexpected:\n%s\n",         \
104             __FILE__, __LINE__, #GOT, format(got), format(want));       \
105     count_error();                                                      \
106   }                                                                     \
107   ++tests;                                                              \
108  } while(0)
109
110 /** @brief Check that a string prefix matches
111  * @param GOT What we actually got
112  * @param WANT What we wanted
113  *
114  * If @p WANT is not a prefix of @p GOT then logs an error (and continues).
115  *
116  * @p WANT is allowed to evaluate to a null pointer (but if it comes to
117  * anything else then must be safe to strcmp).
118  */
119 #define check_string_prefix(GOT, WANT) do {                             \
120   const char *got = GOT;                                                \
121   const char *want = WANT;                                              \
122                                                                         \
123   if(want == 0) {                                                       \
124     fprintf(stderr, "%s:%d: %s returned 0\n",                           \
125             __FILE__, __LINE__, #GOT);                                  \
126     count_error();                                                      \
127   } else if(strncmp(want, got, strlen(want))) {                         \
128     fprintf(stderr, "%s:%d: %s returned:\n%s\nexpected:\n%s...\n",      \
129             __FILE__, __LINE__, #GOT, format(got), format(want));       \
130     count_error();                                                      \
131   }                                                                     \
132   ++tests;                                                              \
133  } while(0)
134
135 /** @brief Check that a pair of integers match.
136  * @param GOT What we actually got
137  * @param WANT What we wanted
138  *
139  * If @p GOT and @p WANT differ then logs an error (and continues).
140  */
141 #define check_integer(GOT, WANT) do {                           \
142   const intmax_t got = GOT, want = WANT;                        \
143   if(got != want) {                                             \
144     fprintf(stderr, "%s:%d: %s returned: %jd  expected: %jd\n", \
145             __FILE__, __LINE__, #GOT, got, want);               \
146     count_error();                                              \
147   }                                                             \
148   ++tests;                                                      \
149 } while(0)
150
151 /** @brief Check that a function calls fatal()
152  * @param WHAT Expression to evaluate
153  *
154  * Evaluates WHAT and if it does not call fatal(), logs an error.  In any case,
155  * continues.  Modifies exitfn() so that fatal() isn't actually fatal.
156  */
157 #define check_fatal(WHAT) do {                                          \
158   void (*const save_exitfn)(int) attribute((noreturn)) = exitfn;        \
159                                                                         \
160   exitfn = test_exitfn;                                                 \
161   if(setjmp(fatal_env) == 0) {                                          \
162     fprintf(stderr, "Expect an error:\n ");                             \
163     (void)(WHAT);                                                       \
164     fprintf(stderr, "\n%s:%d: %s unexpectedly returned\n",              \
165                      __FILE__, __LINE__, #WHAT);                        \
166     count_error();                                                      \
167   }                                                                     \
168   ++tests;                                                              \
169   exitfn = save_exitfn;                                                 \
170 } while(0)
171
172 void count_error(void);
173 const char *format(const char *s);
174 const char *format_utf32(const uint32_t *s);
175 uint32_t *ucs4parse(const char *s);
176 const char *do_printf(const char *fmt, ...);
177 void test_init(int argc, char **argv);
178
179 extern jmp_buf fatal_env;
180 void test_exitfn(int) attribute((noreturn));
181
182 /** @brief Common code for each test source file
183  * @param name Name of test
184  *
185  * Expands to a  @c main function which:
186  * - calls test_init()
187  * - calls test_NAME()
188  * - reports a count of errors
189  * - returns the right exit status
190  */
191 #define TEST(name)                                                      \
192   int main(int argc, char **argv) {                                     \
193     test_init(argc, argv);                                              \
194     test_##name();                                                      \
195     if(errors || verbose)                                               \
196       fprintf(stderr, "test_"#name": %lld errors out of %lld tests\n",  \
197               errors, tests);                                           \
198     if(errors)                                                          \
199       return 1;                                                         \
200     if(skipped)                                                         \
201       return 77;                                                        \
202     return 0;                                                           \
203   }                                                                     \
204                                                                         \
205   struct swallow_semicolon
206
207 #endif /* TEST_H */
208
209 /*
210 Local Variables:
211 c-basic-offset:2
212 comment-column:40
213 fill-column:79
214 indent-tabs-mode:nil
215 End:
216 */