/* * Copyright 2015 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include <vector> #include "buffet/avahi_mdns_client.h" #include <avahi-common/address.h> #include <avahi-common/defs.h> #include <avahi-common/error.h> #include <base/guid.h> #include <brillo/errors/error.h> using brillo::ErrorPtr; namespace buffet { std::unique_ptr<MdnsClient> MdnsClient::CreateInstance() { return std::unique_ptr<MdnsClient>{new AvahiMdnsClient()}; } namespace { void HandleGroupStateChanged(AvahiEntryGroup* g, AvahiEntryGroupState state, AVAHI_GCC_UNUSED void* userdata) { if (state == AVAHI_ENTRY_GROUP_COLLISION || state == AVAHI_ENTRY_GROUP_FAILURE) { LOG(ERROR) << "Avahi service group error: " << state; } } } // namespace AvahiMdnsClient::AvahiMdnsClient() : service_name_(base::GenerateGUID()) { thread_pool_.reset(avahi_threaded_poll_new()); CHECK(thread_pool_); int ret = 0; client_.reset(avahi_client_new( avahi_threaded_poll_get(thread_pool_.get()), {}, &AvahiMdnsClient::OnAvahiClientStateUpdate, this, &ret)); CHECK(client_) << avahi_strerror(ret); avahi_threaded_poll_start(thread_pool_.get()); group_.reset(avahi_entry_group_new(client_.get(), HandleGroupStateChanged, nullptr)); CHECK(group_) << avahi_strerror(avahi_client_errno(client_.get())) << ". Check avahi-daemon configuration"; } AvahiMdnsClient::~AvahiMdnsClient() { if (thread_pool_) avahi_threaded_poll_stop(thread_pool_.get()); } void AvahiMdnsClient::PublishService(const std::string& service_type, uint16_t port, const std::vector<std::string>& txt) { CHECK(group_); CHECK_EQ("_privet._tcp", service_type); if (prev_port_ == port && prev_service_type_ == service_type && txt_records_ == txt) { return; } // Create txt record. std::unique_ptr<AvahiStringList, decltype(&avahi_string_list_free)> txt_list{ nullptr, &avahi_string_list_free}; if (!txt.empty()) { std::vector<const char*> txt_vector_ptr; for (const auto& i : txt) txt_vector_ptr.push_back(i.c_str()); txt_list.reset(avahi_string_list_new_from_array(txt_vector_ptr.data(), txt_vector_ptr.size())); CHECK(txt_list); } int ret = 0; txt_records_ = txt; if (prev_port_ == port && prev_service_type_ == service_type) { ret = avahi_entry_group_update_service_txt_strlst( group_.get(), AVAHI_IF_UNSPEC, AVAHI_PROTO_UNSPEC, {}, service_name_.c_str(), service_type.c_str(), nullptr, txt_list.get()); CHECK_GE(ret, 0) << avahi_strerror(ret); } else { prev_port_ = port; prev_service_type_ = service_type; avahi_entry_group_reset(group_.get()); CHECK(avahi_entry_group_is_empty(group_.get())); ret = avahi_entry_group_add_service_strlst( group_.get(), AVAHI_IF_UNSPEC, AVAHI_PROTO_UNSPEC, {}, service_name_.c_str(), service_type.c_str(), nullptr, nullptr, port, txt_list.get()); CHECK_GE(ret, 0) << avahi_strerror(ret); ret = avahi_entry_group_commit(group_.get()); CHECK_GE(ret, 0) << avahi_strerror(ret); } } void AvahiMdnsClient::StopPublishing(const std::string& service_type) { CHECK(group_); avahi_entry_group_reset(group_.get()); prev_service_type_.clear(); prev_port_ = 0; txt_records_.clear(); } void AvahiMdnsClient::OnAvahiClientStateUpdate(AvahiClient* s, AvahiClientState state, void* userdata) { // Avahi service has been re-initialized (probably due to host name conflict), // so we need to republish the service if it has been previously published. if (state == AVAHI_CLIENT_S_RUNNING) { AvahiMdnsClient* self = static_cast<AvahiMdnsClient*>(userdata); self->RepublishService(); } } void AvahiMdnsClient::RepublishService() { // If we don't have a service to publish, there is nothing else to do here. if (prev_service_type_.empty()) return; LOG(INFO) << "Republishing mDNS service"; std::string service_type = std::move(prev_service_type_); uint16_t port = prev_port_; std::vector<std::string> txt = std::move(txt_records_); StopPublishing(service_type); PublishService(service_type, port, txt); } } // namespace buffet