1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2013 Lennart Poettering
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
30 #include <dbus/dbus.h>
37 #include "bus-message.h"
40 #include "bus-label.h"
42 static void test_bus_path_encode_unique(void) {
43 _cleanup_free_
char *a
= NULL
, *b
= NULL
, *c
= NULL
, *d
= NULL
, *e
= NULL
;
45 assert_se(bus_path_encode_unique(NULL
, "/foo/bar", "some.sender", "a.suffix", &a
) >= 0 && streq_ptr(a
, "/foo/bar/some_2esender/a_2esuffix"));
46 assert_se(bus_path_decode_unique(a
, "/foo/bar", &b
, &c
) > 0 && streq_ptr(b
, "some.sender") && streq_ptr(c
, "a.suffix"));
47 assert_se(bus_path_decode_unique(a
, "/bar/foo", &d
, &d
) == 0 && !d
);
48 assert_se(bus_path_decode_unique("/foo/bar/onlyOneSuffix", "/foo/bar", &d
, &d
) == 0 && !d
);
49 assert_se(bus_path_decode_unique("/foo/bar/_/_", "/foo/bar", &d
, &e
) > 0 && streq_ptr(d
, "") && streq_ptr(e
, ""));
52 static void test_bus_path_encode(void) {
53 _cleanup_free_
char *a
= NULL
, *b
= NULL
, *c
= NULL
, *d
= NULL
, *e
= NULL
, *f
= NULL
;
55 assert_se(sd_bus_path_encode("/foo/bar", "waldo", &a
) >= 0 && streq(a
, "/foo/bar/waldo"));
56 assert_se(sd_bus_path_decode(a
, "/waldo", &b
) == 0 && b
== NULL
);
57 assert_se(sd_bus_path_decode(a
, "/foo/bar", &b
) > 0 && streq(b
, "waldo"));
59 assert_se(sd_bus_path_encode("xxxx", "waldo", &c
) < 0);
60 assert_se(sd_bus_path_encode("/foo/", "waldo", &c
) < 0);
62 assert_se(sd_bus_path_encode("/foo/bar", "", &c
) >= 0 && streq(c
, "/foo/bar/_"));
63 assert_se(sd_bus_path_decode(c
, "/foo/bar", &d
) > 0 && streq(d
, ""));
65 assert_se(sd_bus_path_encode("/foo/bar", "foo.bar", &e
) >= 0 && streq(e
, "/foo/bar/foo_2ebar"));
66 assert_se(sd_bus_path_decode(e
, "/foo/bar", &f
) > 0 && streq(f
, "foo.bar"));
69 static void test_bus_path_encode_many(void) {
70 _cleanup_free_
char *a
= NULL
, *b
= NULL
, *c
= NULL
, *d
= NULL
, *e
= NULL
, *f
= NULL
;
72 assert_se(sd_bus_path_decode_many("/foo/bar", "/prefix/%", NULL
) == 0);
73 assert_se(sd_bus_path_decode_many("/prefix/bar", "/prefix/%bar", NULL
) == 1);
74 assert_se(sd_bus_path_decode_many("/foo/bar", "/prefix/%/suffix", NULL
) == 0);
75 assert_se(sd_bus_path_decode_many("/prefix/foobar/suffix", "/prefix/%/suffix", &a
) == 1 && streq_ptr(a
, "foobar"));
76 assert_se(sd_bus_path_decode_many("/prefix/one_foo_two/mid/three_bar_four/suffix", "/prefix/one_%_two/mid/three_%_four/suffix", &b
, &c
) == 1 && streq_ptr(b
, "foo") && streq_ptr(c
, "bar"));
77 assert_se(sd_bus_path_decode_many("/prefix/one_foo_two/mid/three_bar_four/suffix", "/prefix/one_%_two/mid/three_%_four/suffix", NULL
, &d
) == 1 && streq_ptr(d
, "bar"));
79 assert_se(sd_bus_path_decode_many("/foo/bar", "/foo/bar/%", NULL
) == 0);
80 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/foo/bar%", NULL
) == 0);
81 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/foo/%/bar", NULL
) == 0);
82 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/foo/%bar", NULL
) == 0);
83 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/foo/bar/suffix") == 1);
84 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/foo/%%/suffix", NULL
, NULL
) == 0); /* multiple '%' are treated verbatim */
85 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/foo/%/suffi", NULL
) == 0);
86 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/foo/%/suffix", &e
) == 1 && streq_ptr(e
, "bar"));
87 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/foo/%/%", NULL
, NULL
) == 1);
88 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/%/%/%", NULL
, NULL
, NULL
) == 1);
89 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "%/%/%", NULL
, NULL
, NULL
) == 0);
90 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/%/%", NULL
, NULL
) == 0);
91 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/%/%/", NULL
, NULL
) == 0);
92 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/%/", NULL
) == 0);
93 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "/%", NULL
) == 0);
94 assert_se(sd_bus_path_decode_many("/foo/bar/suffix", "%", NULL
) == 0);
96 assert_se(sd_bus_path_encode_many(&f
, "/prefix/one_%_two/mid/three_%_four/suffix", "foo", "bar") >= 0 && streq_ptr(f
, "/prefix/one_foo_two/mid/three_bar_four/suffix"));
99 static void test_bus_label_escape_one(const char *a
, const char *b
) {
100 _cleanup_free_
char *t
= NULL
, *x
= NULL
, *y
= NULL
;
102 assert_se(t
= bus_label_escape(a
));
103 assert_se(streq(t
, b
));
105 assert_se(x
= bus_label_unescape(t
));
106 assert_se(streq(a
, x
));
108 assert_se(y
= bus_label_unescape(b
));
109 assert_se(streq(a
, y
));
112 static void test_bus_label_escape(void) {
113 test_bus_label_escape_one("foo123bar", "foo123bar");
114 test_bus_label_escape_one("foo.bar", "foo_2ebar");
115 test_bus_label_escape_one("foo_2ebar", "foo_5f2ebar");
116 test_bus_label_escape_one("", "_");
117 test_bus_label_escape_one("_", "_5f");
118 test_bus_label_escape_one("1", "_31");
119 test_bus_label_escape_one(":1", "_3a1");
122 int main(int argc
, char *argv
[]) {
123 _cleanup_bus_message_unref_ sd_bus_message
*m
= NULL
, *copy
= NULL
;
125 const char *x
, *x2
, *y
, *z
, *a
, *b
, *c
, *d
, *a_signature
;
130 const int32_t integer_array
[] = { -1, -2, 0, 1, 2 }, *return_array
;
132 _cleanup_free_
char *first
= NULL
, *second
= NULL
, *third
= NULL
;
133 _cleanup_fclose_
FILE *ms
= NULL
;
134 size_t first_size
= 0, second_size
= 0, third_size
= 0;
135 _cleanup_bus_unref_ sd_bus
*bus
= NULL
;
139 r
= sd_bus_default_system(&bus
);
141 return EXIT_TEST_SKIP
;
143 r
= sd_bus_message_new_method_call(bus
, &m
, "foobar.waldo", "/", "foobar.waldo", "Piep");
146 r
= sd_bus_message_append(m
, "");
149 r
= sd_bus_message_append(m
, "s", "a string");
152 r
= sd_bus_message_append(m
, "s", NULL
);
155 r
= sd_bus_message_append(m
, "asg", 2, "string #1", "string #2", "sba(tt)ss");
158 r
= sd_bus_message_append(m
, "sass", "foobar", 5, "foo", "bar", "waldo", "piep", "pap", "after");
161 r
= sd_bus_message_append(m
, "a{yv}", 2, 3, "s", "foo", 5, "s", "waldo");
164 r
= sd_bus_message_append(m
, "y(ty)y(yt)y", 8, 777ULL, 7, 9, 77, 7777ULL, 10);
167 r
= sd_bus_message_append(m
, "()");
170 r
= sd_bus_message_append(m
, "ba(ss)", 255, 3, "aaa", "1", "bbb", "2", "ccc", "3");
173 r
= sd_bus_message_open_container(m
, 'a', "s");
176 r
= sd_bus_message_append_basic(m
, 's', "foobar");
179 r
= sd_bus_message_append_basic(m
, 's', "waldo");
182 r
= sd_bus_message_close_container(m
);
185 r
= sd_bus_message_append_string_space(m
, 5, &s
);
189 r
= sd_bus_message_append_array(m
, 'i', integer_array
, sizeof(integer_array
));
192 r
= sd_bus_message_append_array(m
, 'u', NULL
, 0);
195 r
= sd_bus_message_append(m
, "a(stdo)", 1, "foo", 815ULL, 47.0, "/");
198 r
= bus_message_seal(m
, 4711, 0);
201 bus_message_dump(m
, stdout
, BUS_MESSAGE_DUMP_WITH_HEADER
);
203 ms
= open_memstream(&first
, &first_size
);
204 bus_message_dump(m
, ms
, 0);
206 assert_se(!ferror(ms
));
208 r
= bus_message_get_blob(m
, &buffer
, &sz
);
211 h
= hexmem(buffer
, sz
);
214 log_info("message size = %zu, contents =\n%s", sz
, h
);
222 #if !defined(GLIB_VERSION_2_36)
226 g
= g_dbus_message_new_from_blob(buffer
, sz
, 0, NULL
);
227 p
= g_dbus_message_print(g
, 0);
239 dbus_error_init(&error
);
241 w
= dbus_message_demarshal(buffer
, sz
, &error
);
243 log_error("%s", error
.message
);
245 dbus_message_unref(w
);
249 m
= sd_bus_message_unref(m
);
251 r
= bus_message_from_malloc(bus
, buffer
, sz
, NULL
, 0, NULL
, &m
);
254 bus_message_dump(m
, stdout
, BUS_MESSAGE_DUMP_WITH_HEADER
);
257 ms
= open_memstream(&second
, &second_size
);
258 bus_message_dump(m
, ms
, 0);
260 assert_se(!ferror(ms
));
261 assert_se(first_size
== second_size
);
262 assert_se(memcmp(first
, second
, first_size
) == 0);
264 assert_se(sd_bus_message_rewind(m
, true) >= 0);
266 r
= sd_bus_message_read(m
, "ssasg", &x
, &x2
, 2, &y
, &z
, &a_signature
);
268 assert_se(streq(x
, "a string"));
269 assert_se(streq(x2
, ""));
270 assert_se(streq(y
, "string #1"));
271 assert_se(streq(z
, "string #2"));
272 assert_se(streq(a_signature
, "sba(tt)ss"));
274 r
= sd_bus_message_read(m
, "sass", &x
, 5, &y
, &z
, &a
, &b
, &c
, &d
);
276 assert_se(streq(x
, "foobar"));
277 assert_se(streq(y
, "foo"));
278 assert_se(streq(z
, "bar"));
279 assert_se(streq(a
, "waldo"));
280 assert_se(streq(b
, "piep"));
281 assert_se(streq(c
, "pap"));
282 assert_se(streq(d
, "after"));
284 r
= sd_bus_message_read(m
, "a{yv}", 2, &u
, "s", &x
, &v
, "s", &y
);
287 assert_se(streq(x
, "foo"));
289 assert_se(streq(y
, "waldo"));
291 r
= sd_bus_message_read(m
, "y(ty)", &v
, &u64
, &u
);
294 assert_se(u64
== 777);
297 r
= sd_bus_message_read(m
, "y(yt)", &v
, &u
, &u64
);
301 assert_se(u64
== 7777);
303 r
= sd_bus_message_read(m
, "y", &v
);
307 r
= sd_bus_message_read(m
, "()");
310 r
= sd_bus_message_read(m
, "ba(ss)", &boolean
, 3, &x
, &y
, &a
, &b
, &c
, &d
);
313 assert_se(streq(x
, "aaa"));
314 assert_se(streq(y
, "1"));
315 assert_se(streq(a
, "bbb"));
316 assert_se(streq(b
, "2"));
317 assert_se(streq(c
, "ccc"));
318 assert_se(streq(d
, "3"));
320 assert_se(sd_bus_message_verify_type(m
, 'a', "s") > 0);
322 r
= sd_bus_message_read(m
, "as", 2, &x
, &y
);
324 assert_se(streq(x
, "foobar"));
325 assert_se(streq(y
, "waldo"));
327 r
= sd_bus_message_read_basic(m
, 's', &s
);
329 assert_se(streq(s
, "hallo"));
331 r
= sd_bus_message_read_array(m
, 'i', (const void**) &return_array
, &sz
);
333 assert_se(sz
== sizeof(integer_array
));
334 assert_se(memcmp(integer_array
, return_array
, sz
) == 0);
336 r
= sd_bus_message_read_array(m
, 'u', (const void**) &return_array
, &sz
);
340 r
= sd_bus_message_read(m
, "a(stdo)", 1, &x
, &u64
, &dbl
, &y
);
342 assert_se(streq(x
, "foo"));
343 assert_se(u64
== 815ULL);
344 assert_se(fabs(dbl
- 47.0) < 0.1);
345 assert_se(streq(y
, "/"));
347 r
= sd_bus_message_peek_type(m
, NULL
, NULL
);
350 r
= sd_bus_message_new_method_call(bus
, ©
, "foobar.waldo", "/", "foobar.waldo", "Piep");
353 r
= sd_bus_message_rewind(m
, true);
356 r
= sd_bus_message_copy(copy
, m
, true);
359 r
= bus_message_seal(copy
, 4712, 0);
363 ms
= open_memstream(&third
, &third_size
);
364 bus_message_dump(copy
, ms
, 0);
366 assert_se(!ferror(ms
));
368 printf("<%.*s>\n", (int) first_size
, first
);
369 printf("<%.*s>\n", (int) third_size
, third
);
371 assert_se(first_size
== third_size
);
372 assert_se(memcmp(first
, third
, third_size
) == 0);
374 r
= sd_bus_message_rewind(m
, true);
377 assert_se(sd_bus_message_verify_type(m
, 's', NULL
) > 0);
379 r
= sd_bus_message_skip(m
, "ssasg");
382 assert_se(sd_bus_message_verify_type(m
, 's', NULL
) > 0);
384 r
= sd_bus_message_skip(m
, "sass");
387 assert_se(sd_bus_message_verify_type(m
, 'a', "{yv}") > 0);
389 r
= sd_bus_message_skip(m
, "a{yv}y(ty)y(yt)y()");
392 assert_se(sd_bus_message_verify_type(m
, 'b', NULL
) > 0);
394 r
= sd_bus_message_read(m
, "b", &boolean
);
398 r
= sd_bus_message_enter_container(m
, 0, NULL
);
401 r
= sd_bus_message_read(m
, "(ss)", &x
, &y
);
404 r
= sd_bus_message_read(m
, "(ss)", &a
, &b
);
407 r
= sd_bus_message_read(m
, "(ss)", &c
, &d
);
410 r
= sd_bus_message_read(m
, "(ss)", &x
, &y
);
413 r
= sd_bus_message_exit_container(m
);
416 assert_se(streq(x
, "aaa"));
417 assert_se(streq(y
, "1"));
418 assert_se(streq(a
, "bbb"));
419 assert_se(streq(b
, "2"));
420 assert_se(streq(c
, "ccc"));
421 assert_se(streq(d
, "3"));
423 test_bus_label_escape();
424 test_bus_path_encode();
425 test_bus_path_encode_unique();
426 test_bus_path_encode_many();