4 This file is part of avahi.
6 avahi is free software; you can redistribute it and/or modify it
7 under the terms of the GNU Lesser General Public License as
8 published by the Free Software Foundation; either version 2.1 of the
9 License, or (at your option) any later version.
11 avahi is distributed in the hope that it will be useful, but WITHOUT
12 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
13 or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General
14 Public License for more details.
16 You should have received a copy of the GNU Lesser General Public
17 License along with avahi; if not, write to the Free Software
18 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
30 #include <dbus/dbus.h>
32 #include <avahi-common/dbus.h>
33 #include <avahi-common/llist.h>
34 #include <avahi-common/error.h>
35 #include <avahi-common/dbus.h>
36 #include <avahi-common/malloc.h>
37 #include <avahi-common/dbus-watch-glue.h>
42 int avahi_client_set_errno (AvahiClient *client, int error) {
45 return client->error = error;
48 int avahi_client_set_dbus_error(AvahiClient *client, DBusError *error) {
52 return avahi_client_set_errno(client, avahi_error_dbus_to_number(error->name));
55 static void client_set_state (AvahiClient *client, AvahiServerState state) {
58 if (client->state == state)
61 client->state = state;
63 switch (client->state) {
64 case AVAHI_CLIENT_DISCONNECTED:
65 case AVAHI_CLIENT_FAILURE:
67 dbus_connection_disconnect(client->bus);
68 dbus_connection_unref(client->bus);
74 case AVAHI_CLIENT_S_COLLISION:
75 case AVAHI_CLIENT_S_REGISTERING:
77 /* Clear cached strings */
78 avahi_free(client->host_name);
79 avahi_free(client->host_name_fqdn);
80 avahi_free(client->domain_name);
82 client->host_name = NULL;
83 client->host_name_fqdn = NULL;
84 client->domain_name = NULL;
87 case AVAHI_CLIENT_S_RUNNING:
93 client->callback (client, state, client->userdata);
96 static DBusHandlerResult filter_func(DBusConnection *bus, DBusMessage *message, void *userdata) {
97 AvahiClient *client = userdata;
103 dbus_error_init (&error);
105 /* fprintf(stderr, "dbus: interface=%s, path=%s, member=%s\n", */
106 /* dbus_message_get_interface (message), */
107 /* dbus_message_get_path (message), */
108 /* dbus_message_get_member (message)); */
110 if (client->state == AVAHI_CLIENT_DISCONNECTED)
113 if (dbus_message_is_signal(message, DBUS_INTERFACE_LOCAL, "Disconnected")) {
115 /* The DBUS server died or kicked us */
116 client_set_state(client, AVAHI_CLIENT_DISCONNECTED);
118 } if (dbus_message_is_signal(message, DBUS_INTERFACE_DBUS, "NameOwnerChanged")) {
119 char *name, *old, *new;
121 if (!dbus_message_get_args(
123 DBUS_TYPE_STRING, &name,
124 DBUS_TYPE_STRING, &old,
125 DBUS_TYPE_STRING, &new,
126 DBUS_TYPE_INVALID) || dbus_error_is_set (&error)) {
128 fprintf(stderr, "WARNING: Failed to parse NameOwnerChanged signal: %s\n", error.message);
132 if (strcmp(name, AVAHI_DBUS_NAME) == 0)
134 /* Regardless if the server lost or acquired its name or
135 * if the name was transfered: our services are no longer
136 * available, so we disconnect ourselves */
138 client_set_state(client, AVAHI_CLIENT_DISCONNECTED);
140 } else if (dbus_message_is_signal (message, AVAHI_DBUS_INTERFACE_SERVER, "StateChanged")) {
145 if (!dbus_message_get_args(
147 DBUS_TYPE_INT32, &state,
148 DBUS_TYPE_STRING, &e,
149 DBUS_TYPE_INVALID) || dbus_error_is_set (&error)) {
150 fprintf(stderr, "WARNING: Failed to parse Server.StateChanged signal: %s\n", error.message);
154 if ((c = avahi_error_dbus_to_number(e)) != AVAHI_OK)
155 avahi_client_set_errno(client, c);
157 client_set_state(client, (AvahiClientState) state);
159 } else if (dbus_message_is_signal (message, AVAHI_DBUS_INTERFACE_ENTRY_GROUP, "StateChanged")) {
162 path = dbus_message_get_path(message);
164 for (g = client->groups; g; g = g->groups_next)
165 if (strcmp(g->path, path) == 0)
173 if (!dbus_message_get_args(
175 DBUS_TYPE_INT32, &state,
176 DBUS_TYPE_STRING, &e,
177 DBUS_TYPE_INVALID) ||
178 dbus_error_is_set(&error)) {
179 fprintf(stderr, "WARNING: Failed to parse EntryGroup.StateChanged signal: %s\n", error.message);
183 if ((c = avahi_error_dbus_to_number(e)) != AVAHI_OK)
184 avahi_client_set_errno(client, c);
186 avahi_entry_group_set_state(g, state);
189 } else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_DOMAIN_BROWSER, "ItemNew"))
190 return avahi_domain_browser_event(client, AVAHI_BROWSER_NEW, message);
191 else if (dbus_message_is_signal (message, AVAHI_DBUS_INTERFACE_DOMAIN_BROWSER, "ItemRemove"))
192 return avahi_domain_browser_event(client, AVAHI_BROWSER_REMOVE, message);
193 else if (dbus_message_is_signal (message, AVAHI_DBUS_INTERFACE_DOMAIN_BROWSER, "CacheExhausted"))
194 return avahi_domain_browser_event(client, AVAHI_BROWSER_CACHE_EXHAUSTED, message);
195 else if (dbus_message_is_signal (message, AVAHI_DBUS_INTERFACE_DOMAIN_BROWSER, "AllForNow"))
196 return avahi_domain_browser_event(client, AVAHI_BROWSER_ALL_FOR_NOW, message);
197 else if (dbus_message_is_signal (message, AVAHI_DBUS_INTERFACE_DOMAIN_BROWSER, "Failure"))
198 return avahi_domain_browser_event(client, AVAHI_BROWSER_FAILURE, message);
200 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_TYPE_BROWSER, "ItemNew"))
201 return avahi_service_type_browser_event (client, AVAHI_BROWSER_NEW, message);
202 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_TYPE_BROWSER, "ItemRemove"))
203 return avahi_service_type_browser_event (client, AVAHI_BROWSER_REMOVE, message);
204 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_TYPE_BROWSER, "CacheExhausted"))
205 return avahi_service_type_browser_event (client, AVAHI_BROWSER_CACHE_EXHAUSTED, message);
206 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_TYPE_BROWSER, "AllForNow"))
207 return avahi_service_type_browser_event (client, AVAHI_BROWSER_ALL_FOR_NOW, message);
208 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_TYPE_BROWSER, "Failure"))
209 return avahi_service_type_browser_event (client, AVAHI_BROWSER_FAILURE, message);
211 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_BROWSER, "ItemNew"))
212 return avahi_service_browser_event (client, AVAHI_BROWSER_NEW, message);
213 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_BROWSER, "ItemRemove"))
214 return avahi_service_browser_event (client, AVAHI_BROWSER_REMOVE, message);
215 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_BROWSER, "CacheExhausted"))
216 return avahi_service_browser_event (client, AVAHI_BROWSER_CACHE_EXHAUSTED, message);
217 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_BROWSER, "AllForNow"))
218 return avahi_service_browser_event (client, AVAHI_BROWSER_ALL_FOR_NOW, message);
219 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_BROWSER, "Failure"))
220 return avahi_service_browser_event (client, AVAHI_BROWSER_FAILURE, message);
222 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_RESOLVER, "Found"))
223 return avahi_service_resolver_event (client, AVAHI_RESOLVER_FOUND, message);
224 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_SERVICE_RESOLVER, "Failure"))
225 return avahi_service_resolver_event (client, AVAHI_RESOLVER_FAILURE, message);
227 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_HOST_NAME_RESOLVER, "Found"))
228 return avahi_host_name_resolver_event (client, AVAHI_RESOLVER_FOUND, message);
229 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_HOST_NAME_RESOLVER, "Failure"))
230 return avahi_host_name_resolver_event (client, AVAHI_RESOLVER_FAILURE, message);
232 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_ADDRESS_RESOLVER, "Found"))
233 return avahi_address_resolver_event (client, AVAHI_RESOLVER_FOUND, message);
234 else if (dbus_message_is_signal(message, AVAHI_DBUS_INTERFACE_ADDRESS_RESOLVER, "Failure"))
235 return avahi_address_resolver_event (client, AVAHI_RESOLVER_FAILURE, message);
237 return DBUS_HANDLER_RESULT_HANDLED;
241 dbus_error_free (&error);
242 return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
245 static int get_server_state(AvahiClient *client, int *ret_error) {
246 DBusMessage *message = NULL, *reply = NULL;
249 int e = AVAHI_ERR_NO_MEMORY;
253 dbus_error_init(&error);
255 if (!(message = dbus_message_new_method_call(AVAHI_DBUS_NAME, AVAHI_DBUS_PATH_SERVER, AVAHI_DBUS_INTERFACE_SERVER, "GetState")))
258 reply = dbus_connection_send_with_reply_and_block (client->bus, message, -1, &error);
260 if (!reply || dbus_error_is_set (&error))
263 if (!dbus_message_get_args(reply, &error, DBUS_TYPE_INT32, &state, DBUS_TYPE_INVALID) ||
264 dbus_error_is_set (&error))
267 client_set_state(client, (AvahiServerState) state);
269 dbus_message_unref(message);
270 dbus_message_unref(reply);
275 if (dbus_error_is_set(&error)) {
276 e = avahi_error_dbus_to_number (error.name);
277 dbus_error_free(&error);
284 dbus_message_unref(message);
286 dbus_message_unref(reply);
291 static int check_version(AvahiClient *client, int *ret_error) {
292 DBusMessage *message = NULL, *reply = NULL;
295 int e = AVAHI_ERR_NO_MEMORY;
299 dbus_error_init(&error);
301 if (!(message = dbus_message_new_method_call(AVAHI_DBUS_NAME, AVAHI_DBUS_PATH_SERVER, AVAHI_DBUS_INTERFACE_SERVER, "GetVersionString")))
304 reply = dbus_connection_send_with_reply_and_block (client->bus, message, -1, &error);
306 if (!reply || dbus_error_is_set (&error))
309 if (!dbus_message_get_args (reply, &error, DBUS_TYPE_STRING, &version, DBUS_TYPE_INVALID) ||
310 dbus_error_is_set (&error))
313 if (strcmp(version, PACKAGE_STRING) != 0) {
314 e = AVAHI_ERR_VERSION_MISMATCH;
318 dbus_message_unref(message);
319 dbus_message_unref(reply);
324 if (dbus_error_is_set(&error)) {
325 e = avahi_error_dbus_to_number (error.name);
326 dbus_error_free(&error);
333 dbus_message_unref(message);
335 dbus_message_unref(reply);
341 /* This function acts like dbus_bus_get but creates a private
342 * connection instead */
343 static DBusConnection*
344 avahi_dbus_bus_get (DBusError *error)
346 DBusConnection *conn;
347 const char *env_addr;
349 env_addr = getenv ("DBUS_SYSTEM_BUS_ADDRESS");
351 if (env_addr == NULL || (*env_addr == 0))
353 env_addr = DBUS_SYSTEM_BUS_DEFAULT_ADDRESS;
356 conn = dbus_connection_open_private (env_addr, error);
361 dbus_connection_set_exit_on_disconnect (conn, FALSE);
363 if (!dbus_bus_register (conn, error))
365 dbus_connection_close (conn);
366 dbus_connection_unref (conn);
374 AvahiClient *avahi_client_new(const AvahiPoll *poll_api, AvahiClientCallback callback, void *userdata, int *ret_error) {
375 AvahiClient *client = NULL;
378 dbus_error_init (&error);
380 if (!(client = avahi_new(AvahiClient, 1))) {
382 *ret_error = AVAHI_ERR_NO_MEMORY;
386 client->poll_api = poll_api;
387 client->error = AVAHI_OK;
388 client->callback = callback;
389 client->userdata = userdata;
390 client->state = AVAHI_CLIENT_DISCONNECTED;
392 client->host_name = NULL;
393 client->host_name_fqdn = NULL;
394 client->domain_name = NULL;
395 client->version_string = NULL;
396 client->local_service_cookie_valid = 0;
398 AVAHI_LLIST_HEAD_INIT(AvahiEntryGroup, client->groups);
399 AVAHI_LLIST_HEAD_INIT(AvahiDomainBrowser, client->domain_browsers);
400 AVAHI_LLIST_HEAD_INIT(AvahiServiceBrowser, client->service_browsers);
401 AVAHI_LLIST_HEAD_INIT(AvahiServiceTypeBrowser, client->service_type_browsers);
402 AVAHI_LLIST_HEAD_INIT(AvahiServiceResolver, client->service_resolvers);
403 AVAHI_LLIST_HEAD_INIT(AvahiHostNameResolver, client->host_name_resolvers);
404 AVAHI_LLIST_HEAD_INIT(AvahiAddressResolver, client->address_resolvers);
406 if (!(client->bus = avahi_dbus_bus_get(&error)) ||
407 dbus_error_is_set (&error))
410 if (avahi_dbus_connection_glue(client->bus, poll_api) < 0) {
412 *ret_error = AVAHI_ERR_NO_MEMORY; /* Not optimal */
416 if (!dbus_connection_add_filter (client->bus, filter_func, client, NULL)) {
418 *ret_error = AVAHI_ERR_NO_MEMORY;
425 "interface='" AVAHI_DBUS_INTERFACE_SERVER "', "
426 "sender='" AVAHI_DBUS_NAME "', "
427 "path='" AVAHI_DBUS_PATH_SERVER "'",
430 if (dbus_error_is_set (&error))
436 "interface='" DBUS_INTERFACE_DBUS "', "
437 "sender='" DBUS_SERVICE_DBUS "', "
438 "path='" DBUS_PATH_DBUS "'",
441 if (dbus_error_is_set (&error))
447 "interface='" DBUS_INTERFACE_LOCAL "'",
450 if (dbus_error_is_set (&error))
453 if (!(dbus_bus_name_has_owner(client->bus, AVAHI_DBUS_NAME, &error)) ||
454 dbus_error_is_set(&error)) {
456 /* We free the error so its not set, that way the fail target
457 * will return the NO_DAEMON error rather than a DBUS error */
458 dbus_error_free (&error);
461 *ret_error = AVAHI_ERR_NO_DAEMON;
466 if (check_version(client, ret_error) < 0)
469 if (get_server_state(client, ret_error) < 0)
477 avahi_client_free(client);
479 if (dbus_error_is_set(&error)) {
482 *ret_error = avahi_error_dbus_to_number(error.name);
484 dbus_error_free(&error);
490 void avahi_client_free(AvahiClient *client) {
493 while (client->groups)
494 avahi_entry_group_free(client->groups);
496 while (client->domain_browsers)
497 avahi_domain_browser_free(client->domain_browsers);
499 while (client->service_browsers)
500 avahi_service_browser_free(client->service_browsers);
502 while (client->service_type_browsers)
503 avahi_service_type_browser_free(client->service_type_browsers);
505 while (client->service_resolvers)
506 avahi_service_resolver_free(client->service_resolvers);
509 dbus_connection_disconnect(client->bus);
510 dbus_connection_unref(client->bus);
513 avahi_free(client->version_string);
514 avahi_free(client->host_name);
515 avahi_free(client->host_name_fqdn);
516 avahi_free(client->domain_name);
521 static char* avahi_client_get_string_reply_and_block (AvahiClient *client, const char *method, const char *param) {
522 DBusMessage *message = NULL, *reply = NULL;
529 dbus_error_init (&error);
531 if (!(message = dbus_message_new_method_call (AVAHI_DBUS_NAME, AVAHI_DBUS_PATH_SERVER, AVAHI_DBUS_INTERFACE_SERVER, method))) {
532 avahi_client_set_errno(client, AVAHI_ERR_NO_MEMORY);
537 if (!dbus_message_append_args (message, DBUS_TYPE_STRING, ¶m, DBUS_TYPE_INVALID)) {
538 avahi_client_set_errno (client, AVAHI_ERR_NO_MEMORY);
543 reply = dbus_connection_send_with_reply_and_block (client->bus, message, -1, &error);
545 if (!reply || dbus_error_is_set (&error))
548 if (!dbus_message_get_args (reply, &error, DBUS_TYPE_STRING, &ret, DBUS_TYPE_INVALID) ||
549 dbus_error_is_set (&error))
552 if (!(n = avahi_strdup(ret))) {
553 avahi_client_set_errno(client, AVAHI_ERR_NO_MEMORY);
557 dbus_message_unref(message);
558 dbus_message_unref(reply);
565 dbus_message_unref(message);
567 dbus_message_unref(reply);
569 if (dbus_error_is_set(&error)) {
570 avahi_client_set_dbus_error(client, &error);
571 dbus_error_free(&error);
577 const char* avahi_client_get_version_string (AvahiClient *client) {
580 if (client->state == AVAHI_CLIENT_DISCONNECTED) {
581 avahi_client_set_errno(client, AVAHI_ERR_BAD_STATE);
585 if (!client->version_string)
586 client->version_string = avahi_client_get_string_reply_and_block(client, "GetVersionString", NULL);
588 return client->version_string;
591 const char* avahi_client_get_domain_name (AvahiClient *client) {
594 if (client->state == AVAHI_CLIENT_DISCONNECTED) {
595 avahi_client_set_errno(client, AVAHI_ERR_BAD_STATE);
599 if (!client->domain_name)
600 client->domain_name = avahi_client_get_string_reply_and_block(client, "GetDomainName", NULL);
602 return client->domain_name;
605 const char* avahi_client_get_host_name (AvahiClient *client) {
608 if (client->state == AVAHI_CLIENT_DISCONNECTED) {
609 avahi_client_set_errno(client, AVAHI_ERR_BAD_STATE);
613 if (!client->host_name)
614 client->host_name = avahi_client_get_string_reply_and_block(client, "GetHostName", NULL);
616 return client->host_name;
619 const char* avahi_client_get_host_name_fqdn (AvahiClient *client) {
622 if (client->state == AVAHI_CLIENT_DISCONNECTED) {
623 avahi_client_set_errno(client, AVAHI_ERR_BAD_STATE);
627 if (!client->host_name_fqdn)
628 client->host_name_fqdn = avahi_client_get_string_reply_and_block(client, "GetHostNameFqdn", NULL);
630 return client->host_name_fqdn;
633 AvahiClientState avahi_client_get_state(AvahiClient *client) {
636 return client->state;
639 int avahi_client_errno(AvahiClient *client) {
642 return client->error;
645 /* Just for internal use */
646 int avahi_client_simple_method_call(AvahiClient *client, const char *path, const char *interface, const char *method) {
647 DBusMessage *message = NULL, *reply = NULL;
651 dbus_error_init(&error);
658 if (!(message = dbus_message_new_method_call(AVAHI_DBUS_NAME, path, interface, method))) {
659 r = avahi_client_set_errno(client, AVAHI_ERR_NO_MEMORY);
663 if (!(reply = dbus_connection_send_with_reply_and_block(client->bus, message, -1, &error)) ||
664 dbus_error_is_set (&error)) {
665 r = avahi_client_set_errno(client, AVAHI_ERR_DBUS_ERROR);
669 if (!dbus_message_get_args(reply, &error, DBUS_TYPE_INVALID) ||
670 dbus_error_is_set (&error)) {
671 r = avahi_client_set_errno(client, AVAHI_ERR_DBUS_ERROR);
675 dbus_message_unref(message);
676 dbus_message_unref(reply);
681 if (dbus_error_is_set(&error)) {
682 r = avahi_client_set_dbus_error(client, &error);
683 dbus_error_free(&error);
687 dbus_message_unref(message);
690 dbus_message_unref(reply);
695 uint32_t avahi_client_get_local_service_cookie(AvahiClient *client) {
696 DBusMessage *message = NULL, *reply = NULL;
700 if (client->state == AVAHI_CLIENT_DISCONNECTED) {
701 avahi_client_set_errno(client, AVAHI_ERR_BAD_STATE);
702 return AVAHI_SERVICE_COOKIE_INVALID;
705 if (client->local_service_cookie_valid)
706 return client->local_service_cookie;
708 dbus_error_init (&error);
710 if (!(message = dbus_message_new_method_call(AVAHI_DBUS_NAME, AVAHI_DBUS_PATH_SERVER, AVAHI_DBUS_INTERFACE_SERVER, "GetLocalServiceCookie"))) {
711 avahi_client_set_errno(client, AVAHI_ERR_NO_MEMORY);
715 reply = dbus_connection_send_with_reply_and_block (client->bus, message, -1, &error);
717 if (!reply || dbus_error_is_set (&error))
720 if (!dbus_message_get_args (reply, &error, DBUS_TYPE_UINT32, &client->local_service_cookie, DBUS_TYPE_INVALID) ||
721 dbus_error_is_set (&error))
724 dbus_message_unref(message);
725 dbus_message_unref(reply);
727 client->local_service_cookie_valid = 1;
728 return client->local_service_cookie;
733 dbus_message_unref(message);
735 dbus_message_unref(reply);
737 if (dbus_error_is_set(&error)) {
738 avahi_client_set_dbus_error(client, &error);
739 dbus_error_free(&error);
742 return AVAHI_SERVICE_COOKIE_INVALID;