lib: make plugin API const-correct
[babeltrace.git] / tests / lib / test_plugin.c
1 /*
2 * Copyright (c) 2017 Philippe Proulx <pproulx@efficios.com>
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; under version 2 of the License.
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License along
14 * with this program; if not, write to the Free Software Foundation, Inc.,
15 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
16 */
17
18 #include <babeltrace/babeltrace.h>
19 #include <stdlib.h>
20 #include <string.h>
21 #include <stdio.h>
22 #include <babeltrace/assert-internal.h>
23 #include <glib.h>
24 #include "tap/tap.h"
25 #include "common.h"
26
27 #define NR_TESTS 35
28 #define NON_EXISTING_PATH "/this/hopefully/does/not/exist/5bc75f8d-0dba-4043-a509-d7984b97e42b.so"
29
30 /* Those symbols are written to by some test plugins */
31 static int check_env_var(const char *name)
32 {
33 const char *val = getenv(name);
34
35 if (!val) {
36 return -1;
37 }
38
39 return atoi(val);
40 }
41
42 static void reset_test_plugin_env_vars(void)
43 {
44 g_setenv("BT_TEST_PLUGIN_INIT_CALLED", "0", 1);
45 g_setenv("BT_TEST_PLUGIN_EXIT_CALLED", "0", 1);
46 }
47
48 static char *get_test_plugin_path(const char *plugin_dir,
49 const char *plugin_name)
50 {
51 char *ret;
52 char *plugin_file_name;
53
54 if (asprintf(&plugin_file_name, "plugin-%s." G_MODULE_SUFFIX,
55 plugin_name) == -1) {
56 abort();
57 }
58
59 ret = g_build_filename(plugin_dir, plugin_file_name, NULL);
60 free(plugin_file_name);
61
62 return ret;
63 }
64
65 static void test_minimal(const char *plugin_dir)
66 {
67 const struct bt_plugin_set *plugin_set;
68 const struct bt_plugin *plugin;
69 char *minimal_path = get_test_plugin_path(plugin_dir, "minimal");
70
71 BT_ASSERT(minimal_path);
72 diag("minimal plugin test below");
73
74 reset_test_plugin_env_vars();
75 plugin_set = bt_plugin_create_all_from_file(minimal_path);
76 ok(plugin_set && bt_plugin_set_get_plugin_count(plugin_set) == 1,
77 "bt_plugin_create_all_from_file() succeeds with a valid file");
78 ok(check_env_var("BT_TEST_PLUGIN_INIT_CALLED") == 1,
79 "plugin's initialization function is called during bt_plugin_create_all_from_file()");
80 ok(bt_plugin_set_get_plugin_count(plugin_set) == 1,
81 "bt_plugin_create_all_from_file() returns the expected number of plugins");
82 plugin = bt_plugin_set_borrow_plugin_by_index_const(plugin_set, 0);
83 ok(strcmp(bt_plugin_get_name(plugin), "test_minimal") == 0,
84 "bt_plugin_get_name() returns the expected name");
85 ok(strcmp(bt_plugin_get_description(plugin),
86 "Minimal Babeltrace plugin with no component classes") == 0,
87 "bt_plugin_get_description() returns the expected description");
88 ok(bt_plugin_get_version(plugin, NULL, NULL, NULL, NULL) ==
89 BT_PROPERTY_AVAILABILITY_NOT_AVAILABLE,
90 "bt_plugin_get_version() fails when there's no version");
91 ok(strcmp(bt_plugin_get_author(plugin), "Janine Sutto") == 0,
92 "bt_plugin_get_author() returns the expected author");
93 ok(strcmp(bt_plugin_get_license(plugin), "Beerware") == 0,
94 "bt_plugin_get_license() returns the expected license");
95 ok(strcmp(bt_plugin_get_path(plugin), minimal_path) == 0,
96 "bt_plugin_get_path() returns the expected path");
97 ok(bt_plugin_get_source_component_class_count(plugin) == 0,
98 "bt_plugin_get_source_component_class_count() returns the expected value");
99 ok(bt_plugin_get_filter_component_class_count(plugin) == 0,
100 "bt_plugin_get_filter_component_class_count() returns the expected value");
101 ok(bt_plugin_get_sink_component_class_count(plugin) == 0,
102 "bt_plugin_get_sink_component_class_count() returns the expected value");
103 bt_object_put_ref(plugin_set);
104 ok(check_env_var("BT_TEST_PLUGIN_EXIT_CALLED") == 1,
105 "plugin's exit function is called when the plugin is destroyed");
106
107 free(minimal_path);
108 }
109
110 static void test_sfs(const char *plugin_dir)
111 {
112 const struct bt_plugin_set *plugin_set;
113 const struct bt_plugin *plugin;
114 struct bt_component_class_sink *sink_comp_class;
115 struct bt_component_class_source *source_comp_class;
116 struct bt_component_class_filter *filter_comp_class;
117 struct bt_component_sink *sink_component;
118 char *sfs_path = get_test_plugin_path(plugin_dir, "sfs");
119 unsigned int major, minor, patch;
120 const char *extra;
121 struct bt_value *params;
122 const struct bt_value *results;
123 const struct bt_value *object;
124 const struct bt_value *res_params;
125 struct bt_private_graph *graph;
126 const char *object_str;
127 enum bt_graph_status graph_ret;
128 struct bt_private_query_executor *query_exec =
129 bt_private_query_executor_create();
130 int ret;
131
132 BT_ASSERT(query_exec);
133 BT_ASSERT(sfs_path);
134 diag("sfs plugin test below");
135
136 plugin_set = bt_plugin_create_all_from_file(sfs_path);
137 BT_ASSERT(plugin_set && bt_plugin_set_get_plugin_count(plugin_set) == 1);
138 plugin = bt_plugin_set_borrow_plugin_by_index_const(plugin_set, 0);
139 ok(bt_plugin_get_version(plugin, &major, &minor, &patch, &extra) ==
140 BT_PROPERTY_AVAILABILITY_AVAILABLE,
141 "bt_plugin_get_version() succeeds when there's a version");
142 ok(major == 1,
143 "bt_plugin_get_version() returns the expected major version");
144 ok(minor == 2,
145 "bt_plugin_get_version() returns the expected minor version");
146 ok(patch == 3,
147 "bt_plugin_get_version() returns the expected patch version");
148 ok(strcmp(extra, "yes") == 0,
149 "bt_plugin_get_version() returns the expected extra version");
150 ok(bt_plugin_get_source_component_class_count(plugin) == 1,
151 "bt_plugin_get_source_component_class_count() returns the expected value");
152 ok(bt_plugin_get_filter_component_class_count(plugin) == 1,
153 "bt_plugin_get_filter_component_class_count() returns the expected value");
154 ok(bt_plugin_get_sink_component_class_count(plugin) == 1,
155 "bt_plugin_get_sink_component_class_count() returns the expected value");
156
157 source_comp_class = bt_plugin_borrow_source_component_class_by_name_const(
158 plugin, "source");
159 ok(source_comp_class,
160 "bt_plugin_borrow_source_component_class_by_name_const() finds a source component class");
161
162 sink_comp_class = bt_plugin_borrow_sink_component_class_by_name_const(
163 plugin, "sink");
164 ok(sink_comp_class,
165 "bt_plugin_borrow_sink_component_class_by_name_const() finds a sink component class");
166 ok(strcmp(bt_component_class_get_help(
167 bt_component_class_sink_as_component_class(sink_comp_class)),
168 "Bacon ipsum dolor amet strip steak cupim pastrami venison shoulder.\n"
169 "Prosciutto beef ribs flank meatloaf pancetta brisket kielbasa drumstick\n"
170 "venison tenderloin cow tail. Beef short loin shoulder meatball, sirloin\n"
171 "ground round brisket salami cupim pork bresaola turkey bacon boudin.\n") == 0,
172 "bt_component_class_get_help() returns the expected help text");
173
174 filter_comp_class = bt_plugin_borrow_filter_component_class_by_name_const(
175 plugin, "filter");
176 ok(filter_comp_class,
177 "bt_plugin_borrow_filter_component_class_by_name_const() finds a filter component class");
178 params = bt_value_integer_create_init(23);
179 BT_ASSERT(params);
180 ret = bt_private_query_executor_query(query_exec,
181 bt_component_class_filter_as_component_class(filter_comp_class),
182 "get-something", params, &results);
183 ok(ret == 0 && results, "bt_private_query_executor_query() succeeds");
184 BT_ASSERT(bt_value_is_array(results) && bt_value_array_get_size(results) == 2);
185 object = bt_value_array_borrow_element_by_index_const(results, 0);
186 BT_ASSERT(object && bt_value_is_string(object));
187 object_str = bt_value_string_get(object);
188 ok(strcmp(object_str, "get-something") == 0,
189 "bt_component_class_query() receives the expected object name");
190 res_params = bt_value_array_borrow_element_by_index_const(results, 1);
191 ok(bt_value_compare(res_params, params),
192 "bt_component_class_query() receives the expected parameters");
193
194 bt_object_get_ref(sink_comp_class);
195 diag("> putting the plugin set object here");
196 BT_OBJECT_PUT_REF_AND_RESET(plugin_set);
197 graph = bt_private_graph_create();
198 BT_ASSERT(graph);
199 graph_ret = bt_private_graph_add_sink_component(graph, sink_comp_class, "the-sink",
200 NULL, &sink_component);
201 ok(graph_ret == BT_GRAPH_STATUS_OK && sink_component,
202 "bt_private_graph_add_sink_component() still works after the plugin object is destroyed");
203 BT_OBJECT_PUT_REF_AND_RESET(sink_component);
204 bt_object_put_ref(graph);
205
206 free(sfs_path);
207 bt_object_put_ref(sink_comp_class);
208 bt_object_put_ref(results);
209 bt_object_put_ref(params);
210 bt_object_put_ref(query_exec);
211 }
212
213 static void test_create_all_from_dir(const char *plugin_dir)
214 {
215 const struct bt_plugin_set *plugin_set;
216
217 diag("create from all test below");
218
219 plugin_set = bt_plugin_create_all_from_dir(NON_EXISTING_PATH, BT_FALSE);
220 ok(!plugin_set,
221 "bt_plugin_create_all_from_dir() fails with an invalid path");
222
223 plugin_set = bt_plugin_create_all_from_dir(plugin_dir, BT_FALSE);
224 ok(plugin_set, "bt_plugin_create_all_from_dir() succeeds with a valid path");
225
226 /* 2 or 4, if `.la` files are considered or not */
227 ok(bt_plugin_set_get_plugin_count(plugin_set) == 2 ||
228 bt_plugin_set_get_plugin_count(plugin_set) == 4,
229 "bt_plugin_create_all_from_dir() returns the expected number of plugin objects");
230
231 bt_object_put_ref(plugin_set);
232 }
233
234 static void test_find(const char *plugin_dir)
235 {
236 int ret;
237 const struct bt_plugin *plugin;
238 char *plugin_path;
239
240 ok(!bt_plugin_find(NON_EXISTING_PATH),
241 "bt_plugin_find() returns NULL with an unknown plugin name");
242 ret = asprintf(&plugin_path, "%s" G_SEARCHPATH_SEPARATOR_S
243 G_DIR_SEPARATOR_S "ec1d09e5-696c-442e-b1c3-f9c6cf7f5958"
244 G_SEARCHPATH_SEPARATOR_S G_SEARCHPATH_SEPARATOR_S
245 G_SEARCHPATH_SEPARATOR_S "%s" G_SEARCHPATH_SEPARATOR_S
246 "8db46494-a398-466a-9649-c765ae077629"
247 G_SEARCHPATH_SEPARATOR_S,
248 NON_EXISTING_PATH, plugin_dir);
249 BT_ASSERT(ret > 0 && plugin_path);
250 g_setenv("BABELTRACE_PLUGIN_PATH", plugin_path, 1);
251 plugin = bt_plugin_find("test_minimal");
252 ok(plugin,
253 "bt_plugin_find() succeeds with a plugin name it can find");
254 ok(strcmp(bt_plugin_get_author(plugin), "Janine Sutto") == 0,
255 "bt_plugin_find() finds the correct plugin for a given name");
256 BT_OBJECT_PUT_REF_AND_RESET(plugin);
257 free(plugin_path);
258 }
259
260 int main(int argc, char **argv)
261 {
262 int ret;
263 const char *plugin_dir;
264
265 if (argc != 2) {
266 puts("Usage: test_plugin plugin_directory");
267 ret = 1;
268 goto end;
269 }
270
271 plugin_dir = argv[1];
272 plan_tests(NR_TESTS);
273 test_minimal(plugin_dir);
274 test_sfs(plugin_dir);
275 test_create_all_from_dir(plugin_dir);
276 test_find(plugin_dir);
277 ret = exit_status();
278 end:
279 return ret;
280 }
This page took 0.035681 seconds and 4 git commands to generate.