]> git.ipfire.org Git - thirdparty/systemd.git/blame - src/libsystemd/sd-bus/test-bus-match.c
tmpfiles: accurately report creation results
[thirdparty/systemd.git] / src / libsystemd / sd-bus / test-bus-match.c
CommitLineData
392d5b37
LP
1/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
2
3/***
4 This file is part of systemd.
5
6 Copyright 2013 Lennart Poettering
7
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.
12
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.
17
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/>.
20***/
21
392d5b37
LP
22#include "log.h"
23#include "util.h"
24#include "macro.h"
25
26#include "bus-match.h"
27#include "bus-message.h"
40ca29a1 28#include "bus-util.h"
19befb2d 29#include "bus-slot.h"
392d5b37
LP
30
31static bool mask[32];
32
ebcf1f97 33static int filter(sd_bus *b, sd_bus_message *m, void *userdata, sd_bus_error *ret_error) {
198b158f 34 log_info("Ran %u", PTR_TO_UINT(userdata));
0c0cdb06 35 assert_se(PTR_TO_UINT(userdata) < ELEMENTSOF(mask));
198b158f 36 mask[PTR_TO_UINT(userdata)] = true;
392d5b37
LP
37 return 0;
38}
39
40static bool mask_contains(unsigned a[], unsigned n) {
41 unsigned i, j;
42
43 for (i = 0; i < ELEMENTSOF(mask); i++) {
44 bool found = false;
45
46 for (j = 0; j < n; j++)
47 if (a[j] == i) {
48 found = true;
49 break;
50 }
51
52 if (found != mask[i])
53 return false;
54 }
55
56 return true;
57}
58
19befb2d 59static int match_add(sd_bus_slot *slots, struct bus_match_node *root, const char *match, int value) {
c7819669
LP
60 struct bus_match_component *components = NULL;
61 unsigned n_components = 0;
19befb2d 62 sd_bus_slot *s;
c7819669
LP
63 int r;
64
19befb2d
LP
65 s = slots + value;
66 zero(*s);
c7819669
LP
67
68 r = bus_match_parse(match, &components, &n_components);
69 if (r < 0)
70 return r;
71
19befb2d
LP
72 s->userdata = INT_TO_PTR(value);
73 s->match_callback.callback = filter;
74
75 r = bus_match_add(root, components, n_components, &s->match_callback);
c7819669
LP
76 bus_match_parse_free(components, n_components);
77
78 return r;
79}
80
392d5b37 81int main(int argc, char *argv[]) {
2a0958d2
LP
82 struct bus_match_node root = {
83 .type = BUS_MATCH_ROOT,
84 };
85
392d5b37 86 _cleanup_bus_message_unref_ sd_bus_message *m = NULL;
198b158f 87 _cleanup_bus_close_unref_ sd_bus *bus = NULL;
392d5b37 88 enum bus_match_node_type i;
198b158f 89 sd_bus_slot slots[19];
2a0958d2 90 int r;
392d5b37 91
2a0958d2
LP
92 r = sd_bus_open_system(&bus);
93 if (r < 0)
94 return EXIT_TEST_SKIP;
392d5b37 95
19befb2d
LP
96 assert_se(match_add(slots, &root, "arg2='wal\\'do',sender='foo',type='signal',interface='bar.x',", 1) >= 0);
97 assert_se(match_add(slots, &root, "arg2='wal\\'do2',sender='foo',type='signal',interface='bar.x',", 2) >= 0);
98 assert_se(match_add(slots, &root, "arg3='test',sender='foo',type='signal',interface='bar.x',", 3) >= 0);
99 assert_se(match_add(slots, &root, "arg3='test',sender='foo',type='method_call',interface='bar.x',", 4) >= 0);
100 assert_se(match_add(slots, &root, "", 5) >= 0);
101 assert_se(match_add(slots, &root, "interface='quux.x'", 6) >= 0);
102 assert_se(match_add(slots, &root, "interface='bar.x'", 7) >= 0);
103 assert_se(match_add(slots, &root, "member='waldo',path='/foo/bar'", 8) >= 0);
104 assert_se(match_add(slots, &root, "path='/foo/bar'", 9) >= 0);
105 assert_se(match_add(slots, &root, "path_namespace='/foo'", 10) >= 0);
106 assert_se(match_add(slots, &root, "path_namespace='/foo/quux'", 11) >= 0);
107 assert_se(match_add(slots, &root, "arg1='two'", 12) >= 0);
108 assert_se(match_add(slots, &root, "member='waldo',arg2path='/prefix/'", 13) >= 0);
109 assert_se(match_add(slots, &root, "member=waldo,path='/foo/bar',arg3namespace='prefix'", 14) >= 0);
198b158f
LP
110 assert_se(match_add(slots, &root, "arg4='pi'", 15) >= 0);
111 assert_se(match_add(slots, &root, "arg4='pa'", 16) >= 0);
112 assert_se(match_add(slots, &root, "arg4='po'", 17) >= 0);
113 assert_se(match_add(slots, &root, "arg4='pu'", 18) >= 0);
392d5b37
LP
114
115 bus_match_dump(&root, 0);
116
2a0958d2 117 assert_se(sd_bus_message_new_signal(bus, &m, "/foo/bar", "bar.x", "waldo") >= 0);
198b158f 118 assert_se(sd_bus_message_append(m, "ssssas", "one", "two", "/prefix/three", "prefix.four", 3, "pi", "pa", "po") >= 0);
3df7a7e6 119 assert_se(bus_message_seal(m, 1, 0) >= 0);
392d5b37
LP
120
121 zero(mask);
eb01ba5d 122 assert_se(bus_match_run(NULL, &root, m) == 0);
198b158f 123 assert_se(mask_contains((unsigned[]) { 9, 8, 7, 5, 10, 12, 13, 14, 15, 16, 17 }, 11));
392d5b37 124
19befb2d
LP
125 assert_se(bus_match_remove(&root, &slots[8].match_callback) >= 0);
126 assert_se(bus_match_remove(&root, &slots[13].match_callback) >= 0);
392d5b37
LP
127
128 bus_match_dump(&root, 0);
129
130 zero(mask);
eb01ba5d 131 assert_se(bus_match_run(NULL, &root, m) == 0);
198b158f 132 assert_se(mask_contains((unsigned[]) { 9, 5, 10, 12, 14, 7, 15, 16, 17 }, 9));
392d5b37
LP
133
134 for (i = 0; i < _BUS_MATCH_NODE_TYPE_MAX; i++) {
135 char buf[32];
136 const char *x;
137
138 assert_se(x = bus_match_node_type_to_string(i, buf, sizeof(buf)));
139
140 if (i >= BUS_MATCH_MESSAGE_TYPE)
141 assert_se(bus_match_node_type_from_string(x, strlen(x)) == i);
142 }
143
144 bus_match_free(&root);
145
146 return 0;
147}