chiark / gitweb /
Prep v239: Add missing updates that evaded migration.
[elogind.git] / src / test / test-string-util.c
1 /* SPDX-License-Identifier: LGPL-2.1+ */
2
3 #include "alloc-util.h"
4 //#include "locale-util.h"
5 #include "macro.h"
6 #include "string-util.h"
7 #include "strv.h"
8 //#include "utf8.h"
9
10 static void test_string_erase(void) {
11         char *x;
12
13         x = strdupa("");
14         assert_se(streq(string_erase(x), ""));
15
16         x = strdupa("1");
17         assert_se(streq(string_erase(x), ""));
18
19         x = strdupa("123456789");
20         assert_se(streq(string_erase(x), ""));
21
22         assert_se(x[1] == '\0');
23         assert_se(x[2] == '\0');
24         assert_se(x[3] == '\0');
25         assert_se(x[4] == '\0');
26         assert_se(x[5] == '\0');
27         assert_se(x[6] == '\0');
28         assert_se(x[7] == '\0');
29         assert_se(x[8] == '\0');
30         assert_se(x[9] == '\0');
31 }
32
33 #if 0 /// UNNEEDED by elogind
34 static void test_ascii_strcasecmp_n(void) {
35
36         assert_se(ascii_strcasecmp_n("", "", 0) == 0);
37         assert_se(ascii_strcasecmp_n("", "", 1) == 0);
38         assert_se(ascii_strcasecmp_n("", "a", 1) < 0);
39         assert_se(ascii_strcasecmp_n("", "a", 2) < 0);
40         assert_se(ascii_strcasecmp_n("a", "", 1) > 0);
41         assert_se(ascii_strcasecmp_n("a", "", 2) > 0);
42         assert_se(ascii_strcasecmp_n("a", "a", 1) == 0);
43         assert_se(ascii_strcasecmp_n("a", "a", 2) == 0);
44         assert_se(ascii_strcasecmp_n("a", "b", 1) < 0);
45         assert_se(ascii_strcasecmp_n("a", "b", 2) < 0);
46         assert_se(ascii_strcasecmp_n("b", "a", 1) > 0);
47         assert_se(ascii_strcasecmp_n("b", "a", 2) > 0);
48         assert_se(ascii_strcasecmp_n("xxxxyxxxx", "xxxxYxxxx", 9) == 0);
49         assert_se(ascii_strcasecmp_n("xxxxxxxxx", "xxxxyxxxx", 9) < 0);
50         assert_se(ascii_strcasecmp_n("xxxxXxxxx", "xxxxyxxxx", 9) < 0);
51         assert_se(ascii_strcasecmp_n("xxxxxxxxx", "xxxxYxxxx", 9) < 0);
52         assert_se(ascii_strcasecmp_n("xxxxXxxxx", "xxxxYxxxx", 9) < 0);
53
54         assert_se(ascii_strcasecmp_n("xxxxYxxxx", "xxxxYxxxx", 9) == 0);
55         assert_se(ascii_strcasecmp_n("xxxxyxxxx", "xxxxxxxxx", 9) > 0);
56         assert_se(ascii_strcasecmp_n("xxxxyxxxx", "xxxxXxxxx", 9) > 0);
57         assert_se(ascii_strcasecmp_n("xxxxYxxxx", "xxxxxxxxx", 9) > 0);
58         assert_se(ascii_strcasecmp_n("xxxxYxxxx", "xxxxXxxxx", 9) > 0);
59 }
60
61 static void test_ascii_strcasecmp_nn(void) {
62         assert_se(ascii_strcasecmp_nn("", 0, "", 0) == 0);
63         assert_se(ascii_strcasecmp_nn("", 0, "", 1) < 0);
64         assert_se(ascii_strcasecmp_nn("", 1, "", 0) > 0);
65         assert_se(ascii_strcasecmp_nn("", 1, "", 1) == 0);
66
67         assert_se(ascii_strcasecmp_nn("aaaa", 4, "aaAa", 4) == 0);
68         assert_se(ascii_strcasecmp_nn("aaa", 3, "aaAa", 4) < 0);
69         assert_se(ascii_strcasecmp_nn("aaa", 4, "aaAa", 4) < 0);
70         assert_se(ascii_strcasecmp_nn("aaaa", 4, "aaA", 3) > 0);
71         assert_se(ascii_strcasecmp_nn("aaaa", 4, "AAA", 4) > 0);
72
73         assert_se(ascii_strcasecmp_nn("aaaa", 4, "bbbb", 4) < 0);
74         assert_se(ascii_strcasecmp_nn("aaAA", 4, "BBbb", 4) < 0);
75         assert_se(ascii_strcasecmp_nn("BBbb", 4, "aaaa", 4) > 0);
76 }
77 #endif // 0
78
79 static void test_cellescape(void) {
80         char buf[40];
81
82         assert_se(streq(cellescape(buf, 1, ""), ""));
83         assert_se(streq(cellescape(buf, 1, "1"), ""));
84         assert_se(streq(cellescape(buf, 1, "12"), ""));
85
86         assert_se(streq(cellescape(buf, 2, ""), ""));
87         assert_se(streq(cellescape(buf, 2, "1"), "1"));
88         assert_se(streq(cellescape(buf, 2, "12"), "."));
89         assert_se(streq(cellescape(buf, 2, "123"), "."));
90
91         assert_se(streq(cellescape(buf, 3, ""), ""));
92         assert_se(streq(cellescape(buf, 3, "1"), "1"));
93         assert_se(streq(cellescape(buf, 3, "12"), "12"));
94         assert_se(streq(cellescape(buf, 3, "123"), ".."));
95         assert_se(streq(cellescape(buf, 3, "1234"), ".."));
96
97         assert_se(streq(cellescape(buf, 4, ""), ""));
98         assert_se(streq(cellescape(buf, 4, "1"), "1"));
99         assert_se(streq(cellescape(buf, 4, "12"), "12"));
100         assert_se(streq(cellescape(buf, 4, "123"), "123"));
101         assert_se(streq(cellescape(buf, 4, "1234"), is_locale_utf8() ? "…" : "..."));
102         assert_se(streq(cellescape(buf, 4, "12345"), is_locale_utf8() ? "…" : "..."));
103
104         assert_se(streq(cellescape(buf, 5, ""), ""));
105         assert_se(streq(cellescape(buf, 5, "1"), "1"));
106         assert_se(streq(cellescape(buf, 5, "12"), "12"));
107         assert_se(streq(cellescape(buf, 5, "123"), "123"));
108         assert_se(streq(cellescape(buf, 5, "1234"), "1234"));
109         assert_se(streq(cellescape(buf, 5, "12345"), is_locale_utf8() ? "1…" : "1..."));
110         assert_se(streq(cellescape(buf, 5, "123456"), is_locale_utf8() ? "1…" : "1..."));
111
112         assert_se(streq(cellescape(buf, 1, "\020"), ""));
113         assert_se(streq(cellescape(buf, 2, "\020"), "."));
114         assert_se(streq(cellescape(buf, 3, "\020"), ".."));
115         assert_se(streq(cellescape(buf, 4, "\020"), "…"));
116         assert_se(streq(cellescape(buf, 5, "\020"), "\\020"));
117
118         assert_se(streq(cellescape(buf, 5, "1234\020"), "1…"));
119         assert_se(streq(cellescape(buf, 6, "1234\020"), "12…"));
120         assert_se(streq(cellescape(buf, 7, "1234\020"), "123…"));
121         assert_se(streq(cellescape(buf, 8, "1234\020"), "1234…"));
122         assert_se(streq(cellescape(buf, 9, "1234\020"), "1234\\020"));
123
124         assert_se(streq(cellescape(buf, 1, "\t\n"), ""));
125         assert_se(streq(cellescape(buf, 2, "\t\n"), "."));
126         assert_se(streq(cellescape(buf, 3, "\t\n"), ".."));
127         assert_se(streq(cellescape(buf, 4, "\t\n"), "…"));
128         assert_se(streq(cellescape(buf, 5, "\t\n"), "\\t\\n"));
129
130         assert_se(streq(cellescape(buf, 5, "1234\t\n"), "1…"));
131         assert_se(streq(cellescape(buf, 6, "1234\t\n"), "12…"));
132         assert_se(streq(cellescape(buf, 7, "1234\t\n"), "123…"));
133         assert_se(streq(cellescape(buf, 8, "1234\t\n"), "1234…"));
134         assert_se(streq(cellescape(buf, 9, "1234\t\n"), "1234\\t\\n"));
135
136         assert_se(streq(cellescape(buf, 4, "x\t\020\n"), "…"));
137         assert_se(streq(cellescape(buf, 5, "x\t\020\n"), "x…"));
138         assert_se(streq(cellescape(buf, 6, "x\t\020\n"), "x…"));
139         assert_se(streq(cellescape(buf, 7, "x\t\020\n"), "x\\t…"));
140         assert_se(streq(cellescape(buf, 8, "x\t\020\n"), "x\\t…"));
141         assert_se(streq(cellescape(buf, 9, "x\t\020\n"), "x\\t…"));
142         assert_se(streq(cellescape(buf, 10, "x\t\020\n"), "x\\t\\020\\n"));
143
144         assert_se(streq(cellescape(buf, 6, "1\011"), "1\\t"));
145         assert_se(streq(cellescape(buf, 6, "1\020"), "1\\020"));
146         assert_se(streq(cellescape(buf, 6, "1\020x"), is_locale_utf8() ? "1…" : "1..."));
147
148         assert_se(streq(cellescape(buf, 40, "1\020"), "1\\020"));
149         assert_se(streq(cellescape(buf, 40, "1\020x"), "1\\020x"));
150
151         assert_se(streq(cellescape(buf, 40, "\a\b\f\n\r\t\v\\\"'"), "\\a\\b\\f\\n\\r\\t\\v\\\\\\\"\\'"));
152         assert_se(streq(cellescape(buf, 6, "\a\b\f\n\r\t\v\\\"'"), is_locale_utf8() ? "\\a…" : "\\a..."));
153         assert_se(streq(cellescape(buf, 7, "\a\b\f\n\r\t\v\\\"'"), is_locale_utf8() ? "\\a…" : "\\a..."));
154         assert_se(streq(cellescape(buf, 8, "\a\b\f\n\r\t\v\\\"'"), is_locale_utf8() ? "\\a\\b…" : "\\a\\b..."));
155
156         assert_se(streq(cellescape(buf, sizeof buf, "1\020"), "1\\020"));
157         assert_se(streq(cellescape(buf, sizeof buf, "1\020x"), "1\\020x"));
158 }
159
160 static void test_streq_ptr(void) {
161         assert_se(streq_ptr(NULL, NULL));
162         assert_se(!streq_ptr("abc", "cdef"));
163 }
164
165 static void test_strstrip(void) {
166         char *r;
167         char input[] = "   hello, waldo.   ";
168
169         r = strstrip(input);
170         assert_se(streq(r, "hello, waldo."));
171 }
172
173 static void test_strextend(void) {
174         _cleanup_free_ char *str = NULL;
175
176         assert_se(strextend(&str, NULL));
177         assert_se(streq_ptr(str, ""));
178         assert_se(strextend(&str, "", "0", "", "", "123", NULL));
179         assert_se(streq_ptr(str, "0123"));
180         assert_se(strextend(&str, "456", "78", "9", NULL));
181         assert_se(streq_ptr(str, "0123456789"));
182 }
183
184 static void test_strextend_with_separator(void) {
185         _cleanup_free_ char *str = NULL;
186
187         assert_se(strextend_with_separator(&str, NULL, NULL));
188         assert_se(streq_ptr(str, ""));
189         str = mfree(str);
190
191         assert_se(strextend_with_separator(&str, "...", NULL));
192         assert_se(streq_ptr(str, ""));
193         assert_se(strextend_with_separator(&str, "...", NULL));
194         assert_se(streq_ptr(str, ""));
195         str = mfree(str);
196
197         assert_se(strextend_with_separator(&str, "xyz", "a", "bb", "ccc", NULL));
198         assert_se(streq_ptr(str, "axyzbbxyzccc"));
199         str = mfree(str);
200
201         assert_se(strextend_with_separator(&str, ",", "start", "", "1", "234", NULL));
202         assert_se(streq_ptr(str, "start,,1,234"));
203         assert_se(strextend_with_separator(&str, ";", "more", "5", "678", NULL));
204         assert_se(streq_ptr(str, "start,,1,234;more;5;678"));
205 }
206
207 static void test_strrep(void) {
208         _cleanup_free_ char *one, *three, *zero;
209         one = strrep("waldo", 1);
210         three = strrep("waldo", 3);
211         zero = strrep("waldo", 0);
212
213         assert_se(streq(one, "waldo"));
214         assert_se(streq(three, "waldowaldowaldo"));
215         assert_se(streq(zero, ""));
216 }
217
218 static void test_strappend(void) {
219         _cleanup_free_ char *t1, *t2, *t3, *t4;
220
221         t1 = strappend(NULL, NULL);
222         assert_se(streq(t1, ""));
223
224         t2 = strappend(NULL, "suf");
225         assert_se(streq(t2, "suf"));
226
227         t3 = strappend("pre", NULL);
228         assert_se(streq(t3, "pre"));
229
230         t4 = strappend("pre", "suf");
231         assert_se(streq(t4, "presuf"));
232 }
233
234 static void test_string_has_cc(void) {
235         assert_se(string_has_cc("abc\1", NULL));
236         assert_se(string_has_cc("abc\x7f", NULL));
237         assert_se(string_has_cc("abc\x7f", NULL));
238         assert_se(string_has_cc("abc\t\x7f", "\t"));
239         assert_se(string_has_cc("abc\t\x7f", "\t"));
240         assert_se(string_has_cc("\x7f", "\t"));
241         assert_se(string_has_cc("\x7f", "\t\a"));
242
243         assert_se(!string_has_cc("abc\t\t", "\t"));
244         assert_se(!string_has_cc("abc\t\t\a", "\t\a"));
245         assert_se(!string_has_cc("a\ab\tc", "\t\a"));
246 }
247
248 #if 0 /// UNNEEDED by elogind
249 static void test_ascii_strlower(void) {
250         char a[] = "AabBcC Jk Ii Od LKJJJ kkd LK";
251         assert_se(streq(ascii_strlower(a), "aabbcc jk ii od lkjjj kkd lk"));
252 }
253 #endif // 0
254
255 static void test_strshorten(void) {
256         char s[] = "foobar";
257
258         assert_se(strlen(strshorten(s, 6)) == 6);
259         assert_se(strlen(strshorten(s, 12)) == 6);
260         assert_se(strlen(strshorten(s, 2)) == 2);
261         assert_se(strlen(strshorten(s, 0)) == 0);
262 }
263
264 static void test_strjoina(void) {
265         char *actual;
266
267         actual = strjoina("", "foo", "bar");
268         assert_se(streq(actual, "foobar"));
269
270         actual = strjoina("foo", "bar", "baz");
271         assert_se(streq(actual, "foobarbaz"));
272
273         actual = strjoina("foo", "", "bar", "baz");
274         assert_se(streq(actual, "foobarbaz"));
275
276         actual = strjoina("foo");
277         assert_se(streq(actual, "foo"));
278
279         actual = strjoina(NULL);
280         assert_se(streq(actual, ""));
281
282         actual = strjoina(NULL, "foo");
283         assert_se(streq(actual, ""));
284
285         actual = strjoina("foo", NULL, "bar");
286         assert_se(streq(actual, "foo"));
287 }
288
289 static void test_strcmp_ptr(void) {
290         assert_se(strcmp_ptr(NULL, NULL) == 0);
291         assert_se(strcmp_ptr("", NULL) > 0);
292         assert_se(strcmp_ptr("foo", NULL) > 0);
293         assert_se(strcmp_ptr(NULL, "") < 0);
294         assert_se(strcmp_ptr(NULL, "bar") < 0);
295         assert_se(strcmp_ptr("foo", "bar") > 0);
296         assert_se(strcmp_ptr("bar", "baz") < 0);
297         assert_se(strcmp_ptr("foo", "foo") == 0);
298         assert_se(strcmp_ptr("", "") == 0);
299 }
300
301 static void test_foreach_word(void) {
302         const char *word, *state;
303         size_t l;
304         int i = 0;
305         const char test[] = "test abc d\te   f   ";
306         const char * const expected[] = {
307                 "test",
308                 "abc",
309                 "d",
310                 "e",
311                 "f",
312                 "",
313                 NULL
314         };
315
316         FOREACH_WORD(word, l, test, state)
317                 assert_se(strneq(expected[i++], word, l));
318 }
319
320 static void check(const char *test, char** expected, bool trailing) {
321         int i = 0, r;
322
323         printf("<<<%s>>>\n", test);
324         for (;;) {
325                 _cleanup_free_ char *word = NULL;
326
327                 r = extract_first_word(&test, &word, NULL, EXTRACT_QUOTES);
328                 if (r == 0) {
329                         assert_se(!trailing);
330                         break;
331                 } else if (r < 0) {
332                         assert_se(trailing);
333                         break;
334                 }
335
336                 assert_se(streq(word, expected[i++]));
337                 printf("<%s>\n", word);
338         }
339         assert_se(expected[i] == NULL);
340 }
341
342 static void test_foreach_word_quoted(void) {
343         check("test a b c 'd' e '' '' hhh '' '' \"a b c\"",
344               STRV_MAKE("test",
345                         "a",
346                         "b",
347                         "c",
348                         "d",
349                         "e",
350                         "",
351                         "",
352                         "hhh",
353                         "",
354                         "",
355                         "a b c"),
356               false);
357
358         check("test \"xxx",
359               STRV_MAKE("test"),
360               true);
361
362         check("test\\",
363               STRV_MAKE_EMPTY,
364               true);
365 }
366
367 static void test_endswith(void) {
368         assert_se(endswith("foobar", "bar"));
369         assert_se(endswith("foobar", ""));
370         assert_se(endswith("foobar", "foobar"));
371         assert_se(endswith("", ""));
372
373         assert_se(!endswith("foobar", "foo"));
374         assert_se(!endswith("foobar", "foobarfoofoo"));
375 }
376
377 static void test_endswith_no_case(void) {
378         assert_se(endswith_no_case("fooBAR", "bar"));
379         assert_se(endswith_no_case("foobar", ""));
380         assert_se(endswith_no_case("foobar", "FOOBAR"));
381         assert_se(endswith_no_case("", ""));
382
383         assert_se(!endswith_no_case("foobar", "FOO"));
384         assert_se(!endswith_no_case("foobar", "FOOBARFOOFOO"));
385 }
386
387 #if 0 /// UNNEEDED by elogind
388 static void test_delete_chars(void) {
389         char *s, input[] = "   hello, waldo.   abc";
390
391         s = delete_chars(input, WHITESPACE);
392         assert_se(streq(s, "hello,waldo.abc"));
393         assert_se(s == input);
394 }
395 #endif // 0
396
397 static void test_delete_trailing_chars(void) {
398
399         char *s,
400                 input1[] = " \n \r k \n \r ",
401                 input2[] = "kkkkthiskkkiskkkaktestkkk",
402                 input3[] = "abcdef";
403
404         s = delete_trailing_chars(input1, WHITESPACE);
405         assert_se(streq(s, " \n \r k"));
406         assert_se(s == input1);
407
408         s = delete_trailing_chars(input2, "kt");
409         assert_se(streq(s, "kkkkthiskkkiskkkaktes"));
410         assert_se(s == input2);
411
412         s = delete_trailing_chars(input3, WHITESPACE);
413         assert_se(streq(s, "abcdef"));
414         assert_se(s == input3);
415
416         s = delete_trailing_chars(input3, "fe");
417         assert_se(streq(s, "abcd"));
418         assert_se(s == input3);
419 }
420
421 static void test_delete_trailing_slashes(void) {
422         char s1[] = "foobar//",
423              s2[] = "foobar/",
424              s3[] = "foobar",
425              s4[] = "";
426
427         assert_se(streq(delete_trailing_chars(s1, "_"), "foobar//"));
428         assert_se(streq(delete_trailing_chars(s1, "/"), "foobar"));
429         assert_se(streq(delete_trailing_chars(s2, "/"), "foobar"));
430         assert_se(streq(delete_trailing_chars(s3, "/"), "foobar"));
431         assert_se(streq(delete_trailing_chars(s4, "/"), ""));
432 }
433
434 static void test_skip_leading_chars(void) {
435         char input1[] = " \n \r k \n \r ",
436                 input2[] = "kkkkthiskkkiskkkaktestkkk",
437                 input3[] = "abcdef";
438
439         assert_se(streq(skip_leading_chars(input1, WHITESPACE), "k \n \r "));
440         assert_se(streq(skip_leading_chars(input2, "k"), "thiskkkiskkkaktestkkk"));
441         assert_se(streq(skip_leading_chars(input2, "tk"), "hiskkkiskkkaktestkkk"));
442         assert_se(streq(skip_leading_chars(input3, WHITESPACE), "abcdef"));
443         assert_se(streq(skip_leading_chars(input3, "bcaef"), "def"));
444 }
445
446 static void test_in_charset(void) {
447         assert_se(in_charset("dddaaabbbcccc", "abcd"));
448         assert_se(!in_charset("dddaaabbbcccc", "abc f"));
449 }
450
451 static void test_split_pair(void) {
452         _cleanup_free_ char *a = NULL, *b = NULL;
453
454         assert_se(split_pair("", "", &a, &b) == -EINVAL);
455         assert_se(split_pair("foo=bar", "", &a, &b) == -EINVAL);
456         assert_se(split_pair("", "=", &a, &b) == -EINVAL);
457         assert_se(split_pair("foo=bar", "=", &a, &b) >= 0);
458         assert_se(streq(a, "foo"));
459         assert_se(streq(b, "bar"));
460         free(a);
461         free(b);
462         assert_se(split_pair("==", "==", &a, &b) >= 0);
463         assert_se(streq(a, ""));
464         assert_se(streq(b, ""));
465         free(a);
466         free(b);
467
468         assert_se(split_pair("===", "==", &a, &b) >= 0);
469         assert_se(streq(a, ""));
470         assert_se(streq(b, "="));
471 }
472
473 static void test_first_word(void) {
474         assert_se(first_word("Hello", ""));
475         assert_se(first_word("Hello", "Hello"));
476         assert_se(first_word("Hello world", "Hello"));
477         assert_se(first_word("Hello\tworld", "Hello"));
478         assert_se(first_word("Hello\nworld", "Hello"));
479         assert_se(first_word("Hello\rworld", "Hello"));
480         assert_se(first_word("Hello ", "Hello"));
481
482         assert_se(!first_word("Hello", "Hellooo"));
483         assert_se(!first_word("Hello", "xxxxx"));
484         assert_se(!first_word("Hellooo", "Hello"));
485 }
486
487 static void test_strlen_ptr(void) {
488         assert_se(strlen_ptr("foo") == 3);
489         assert_se(strlen_ptr("") == 0);
490         assert_se(strlen_ptr(NULL) == 0);
491 }
492
493 static void test_memory_startswith(void) {
494         assert_se(streq(memory_startswith("", 0, ""), ""));
495         assert_se(streq(memory_startswith("", 1, ""), ""));
496         assert_se(streq(memory_startswith("x", 2, ""), "x"));
497         assert_se(!memory_startswith("", 1, "x"));
498         assert_se(!memory_startswith("", 1, "xxxxxxxx"));
499         assert_se(streq(memory_startswith("xxx", 4, "x"), "xx"));
500         assert_se(streq(memory_startswith("xxx", 4, "xx"), "x"));
501         assert_se(streq(memory_startswith("xxx", 4, "xxx"), ""));
502         assert_se(!memory_startswith("xxx", 4, "xxxx"));
503 }
504
505 int main(int argc, char *argv[]) {
506         test_string_erase();
507 #if 0 /// UNNEEDED by elogind
508         test_ascii_strcasecmp_n();
509         test_ascii_strcasecmp_nn();
510 #endif // 0
511         test_cellescape();
512         test_streq_ptr();
513         test_strstrip();
514         test_strextend();
515         test_strextend_with_separator();
516         test_strrep();
517         test_strappend();
518         test_string_has_cc();
519 #if 0 /// UNNEEDED by elogind
520         test_ascii_strlower();
521 #endif // 0
522         test_strshorten();
523         test_strjoina();
524         test_strcmp_ptr();
525         test_foreach_word();
526         test_foreach_word_quoted();
527         test_endswith();
528         test_endswith_no_case();
529 #if 0 /// UNNEEDED by elogind
530         test_delete_chars();
531 #endif // 0
532         test_delete_trailing_chars();
533         test_delete_trailing_slashes();
534         test_skip_leading_chars();
535         test_in_charset();
536         test_split_pair();
537         test_first_word();
538         test_strlen_ptr();
539         test_memory_startswith();
540
541         return 0;
542 }