# icediscovery ## Introduction This example demonstrates how to search for specific services using iceoryx's `ServiceDiscovery`. It provides two applications - one offering different services and one searching for these making different search queries. A `service` in iceoryx is defined by a `ServiceDescription` that represents a topic under which publisher/server and subscriber/client can exchange data. In addition the applications `iox-wait-for-service` and `iox-discovery-monitor` demonstrate how to write custom discovery functionality to wait for specific services or monitor the availability of services respectively. ## Expected Output [![asciicast](https://asciinema.org/a/476673.svg)](https://asciinema.org/a/476673) ## Code walkthrough ### Offer services We create several publishers which offer their services on construction by default. For more dynamism the `cameraPublishers` offer/stop their services periodically. If you want more information on how to create a publisher, have a look at the [icehello](https://github.com/eclipse-iceoryx/iceoryx/tree/v2.0.0/iceoryx_examples/icehello), [icedelivery](https://github.com/eclipse-iceoryx/iceoryx/tree/v2.0.0/iceoryx_examples/icedelivery), and [iceoptions](https://github.com/eclipse-iceoryx/iceoryx/tree/v2.0.0/iceoryx_examples/iceoptions) examples. ### Find services To be able to search for services, we have to create a `ServiceDiscovery` object: ```cpp iox::runtime::ServiceDiscovery serviceDiscovery; ``` It is included via: ```cpp #include "iceoryx_posh/runtime/service_discovery.hpp" ``` On that object we can call the method `findService` which expects the three service [string identifiers](https://github.com/eclipse-iceoryx/iceoryx/blob/v2.0.0/doc/website/getting-started/overview.md#creating-service-descriptions-for-topics) and a callable which will be applied to all matching services. In addition we have to specify whether we want to search for publishers (`MessagingPattern::PUB_SUB`) used in publish-subscribe communication or servers (`MessagingPattern::REQ_RES`) used in request-response communication. In this example we pass a function that prints the found services on the console: ```cpp void printSearchResult(const iox::capro::ServiceDescription& service) { std::cout << "- " << service << std::endl; } ``` We can search for exactly matching services: ```cpp serviceDiscovery.findService(iox::capro::IdString_t{"Radar"}, iox::capro::IdString_t{"FrontLeft"}, iox::capro::IdString_t{"Image"}, printSearchResult, iox::popo::MessagingPattern::PUB_SUB); ``` or add wildcards to our search query: ```cpp serviceDiscovery.findService(iox::capro::IdString_t{"Camera"}, iox::capro::Wildcard, iox::capro::Wildcard, printSearchResult, iox::popo::MessagingPattern::PUB_SUB); ``` With the above `findService` call we look for every `Camera` service with any instance and any event. Since the `cameraPublishers` periodically offer/stop their services, you should see sometimes 5 `Camera` services and sometimes none. ### Wait for services Start the applications `iox-wait-for-service` and `iox-offer-service`. This can be done in any order, but for demonstration purposes `iox-offer-service` should be started last. `iox-wait-for-service` uses a customized service discovery [Discovery](#implementation-of-discovery-with-blocking-wait) which supports to wait for services by including ```cpp #include "discovery_blocking.hpp" ``` We then can use our custom discovery class ```cpp // requires the runtime to be created first Discovery discovery; ``` which provides a function `waitUntil` to wait for some discovery-related search query condition. We define the search query ```cpp auto query = [&]() { auto result = discovery.findService(service, instance, event); return !result.empty(); }; ``` This is essentially any callable with `bool(void)` signature, but it should depend on the discovery somehow (by capture), as it is only checked when the service availability changes in some way. Here we require some specific service to be found before we proceed. ```cpp iox::capro::IdString_t service{"Camera"}; iox::capro::IdString_t instance{"FrontLeft"}; iox::capro::IdString_t event{"Image"}; ``` Now we can wait until the service discovery changes and the service becomes available. ```cpp bool serviceWasAvailable = discovery.waitUntil(query); ``` This wait is blocking until the service was available. If it already is available we do not block and proceed. It is important that due to the nature of concurrent systems we cannot know that the service is still available once we return from `waitUntil`, as the application offering the service may have stopped doing so in the meantime. Usually we will assume that the service is available and may continue, e.g. by creating subscribers and running application specific code. We can also block until any unspecified change in the service availability occurs ```cpp discovery.waitUntilChange(); ``` This change is relative to the last `findService` call we issued, i.e. if something changed compared to the available services at this point, we wake up and continue. We then can check any condition we like, but usually it will be most useful to again check discovery-related conditions. Here we check whether a particular service becomes unavailable (essentially the negation of our query before) ```cpp if (discovery.findService(service, instance, event).empty()) { break; } ``` Note that we use a customized `findService` version which returns a result container which can easily be build using the version which takes a function to be applied to all services in the search result. Once the service becomes unavailable, the application exits. Should the service we wait for never become available we can unblock any of the wait calls with ```cpp if (discoveryPtr) { discoveryPtr->unblockWait(); } ``` ### Monitor service availability If we want to continously monitor the availability of some service or check some discovery condition we can do so by using e.g. a listener to conditionally execute [callbacks](https://github.com/eclipse-iceoryx/iceoryx/tree/v2.0.0/iceoryx_examples/callbacks). To do so, we start the applications `iox-discovery-monitor` and `iox-offer-service` (again in any order, but for demonstration purposes `iox-offer-service` should be started last). Again we can use a [Discovery](#implementation-of-discovery-monitoring) customized for this purpose by including ```cpp #include "discovery_monitor.hpp" ``` and creating it like so ```cpp // requires the runtime to be created first Discovery discovery; ``` Afterwards we create a callback to be called whenever the service availability changes. ```cpp auto callback = [&](Discovery& discovery) -> void { auto result = discovery.findService(service, instance, event); if (!result.empty()) { std::cout << APP_NAME << " <" << service << ", " << instance << ", " << event << "> available" << std::endl; } else { std::cout << APP_NAME << " <" << service << ", " << instance << ", " << event << "> unavailable" << std::endl; } printSearchResult(result); }; ``` This callback essentially checks whether a specific service is available or unavailable and generates output accordingly. Other reactions are possible as well, such as changing the processing logic of an pplication. To start the monitoring, we register the callback ```cpp discovery.registerCallback(callback); ``` Monitoring happens in a background thread implicitly created by the `Discovery`, i.e. the callback is executed in this thread. When we want to stop monitoring we have to deregister the callback ```cpp discovery.deregisterCallback(); ``` Here this is done at the very end where it is technically not required, but in a more complex application it could be done while the application is processing data. The main processing loop of the application is deliberately left empty for simplicty. Usually it would interact with the callback by e.g. changing application behavior whenever the availability of some service changes. While we only can attach one callback to the general event that the service availability changes in some way, we can generalize the mechanism here to check for multiple conditions and react to each of them by e.g. calling a specific function. These conditions would still need to be checked in the callback we defined though. ### Implementation of Discovery with blocking wait We build our custom discovery on top of the `iox::runtime::ServiceDiscovery` by composition. While inheritance is an option, composition has the advantage that we can use `ServiceDiscovery` as a singleton (to save memory) but our custom `Discovery` class can be fairly lightweight and does not need to be a singleton. ```cpp ServiceDiscovery& serviceDiscovery() { static ServiceDiscovery instance; return instance; } ``` This is useful as the `ServiceDiscovery` may be fairly large and in general there is no point in having multiple `ServiceDiscovery` objects that all have the same purpose and (if updated) same view of the available services. The key idea is to use a waitset and attach to the event that the service availability changes ```cpp m_waitset.attachEvent(*m_discovery, iox::runtime::ServiceDiscoveryEvent::SERVICE_REGISTRY_CHANGED) .or_else(errorHandler); ``` Waiting for any availability change is now as simple as waiting on the waitset ```cpp void Discovery::waitUntilChange() { do { auto notificationVector = m_waitset.wait(); for (auto& notification : notificationVector) { if (notification->doesOriginateFrom(m_discovery)) { return; } } } while (m_blocking); } ``` If we want to wait for a specific condition, we can do so with ```cpp template bool Discovery::waitUntil(const Condition& condition) { do { // 1) does the condition hold? bool result = condition(); if (result) { // 2) condition held and we return (without mutex to protect condition changes // there is no way to guarantee it still holds) return true; } else { if (!m_blocking) { return false; } } // 3) condition did not hold but it may hold if we use the latest discovery data // which may have arrived in the meantime // 4) this does not wait if there is new discovery data (and hence we try again immediately) waitUntilChange(); // 5) discovery data changed, check condition again (even if unblocked) } while (true); return false; } ``` The condition needs to be evaluable to `bool` and takes no arguments. While this can be generalized to any variadic arguments, it is not needed as we can use capturing lambda expressions. The wait simply checks for the condition, and if true returns immediately. Otherwise it waits until the available services change using `waitUntilChange` before checking the condition again. It is also possible to unblock any of the waits even if nothing changes or the condition does not hold ```cpp void Discovery::unblockWait() { m_blocking = false; // could also unblock with a dedicated condition to unblock the wait but that requires more code // (additional trigger) and is not necessary if it is only supposed to happen once m_waitset.markForDestruction(); } ``` This is can only be called once and makes all future wait calls non-blocking. It is useful to unblock any wait calls to be able to stop the application. Finally we provide a custom implementation of `findService` which returns a container of our choice, in this case a `std::vector`. ```cpp ServiceContainer Discovery::findService(const iox::cxx::optional& service, const iox::cxx::optional& instance, const iox::cxx::optional& event) { ServiceContainer result; auto filter = [&](const iox::capro::ServiceDescription& s) { result.emplace_back(s); }; m_discovery->findService(service, instance, event, filter, iox::popo::MessagingPattern::PUB_SUB); return result; } ``` It is implemenented by using the native `findService` call of the `ServiceDiscovery` with an appropriate filter function. The benefit is that this way we can choose containers which do not necessrily reside on the stack. ### Implementation of Discovery monitoring To implement a `Discovery` where we actively monitor availability of services we employ a [listener](https://github.com/eclipse-iceoryx/iceoryx/tree/v2.0.0/iceoryx_examples/callbacks). Contrary to the blocking solution this does not block the user threads and executes any callback in a background thread created by the listener. The callback will be executed on any change of the available services. To register the callback we call ```cpp template bool Discovery::registerCallback(const Callback& callback) ``` which attaches the callback to the listener. ```cpp auto invoker = iox::popo::createNotificationCallback(invokeCallback, *this); m_listener.attachEvent(*m_discovery, iox::runtime::ServiceDiscoveryEvent::SERVICE_REGISTRY_CHANGED, invoker) .or_else(errorHandler); ``` The callback is stored as a `cxx::function` which does not require dynamic memory (but limits the size of the stored function, which is relvant e.g. for capturing lambdas). If dynamic memory is no concern we can also use a `std::function`. The callback can be any callable with a `(void)(discovery::Discovery&)` signature. Again the callback signature can be generalized somewhat but there are constraints to use it with the listener. Since the listener can only call static or free functions, we use an additional indirection to call the actual callback ```cpp void Discovery::invokeCallback(ServiceDiscovery*, Discovery* self) { // discarded discovery argument is required by the listener self->m_callback(*self); } ``` As soon as the callback is registered, the listener thread will invoke it on any service availability change. There is a small caveat though that while callback is called on any change, we can only access the latest discovery information by e.g. calling `findService`. This means all intermediate changes cannot be detected, in particular we may encounter an ABA problem of service availabilty: the service is availalable, becomes unavailable and available again in quick succession. If the callback issues a `findService`, it will not observe any change in this case. As one is usually mainly interested in the available services this can be considered a minor limitation. To stop monitoring changes in the availability of services we simply call ```cpp void Discovery::deregisterCallback() { if (m_callback) { m_listener.detachEvent(*m_discovery, iox::runtime::ServiceDiscoveryEvent::SERVICE_REGISTRY_CHANGED); } m_callback = nullptr; } ``` which detaches the callback from the listener. As before we built on an `iox::runtime::ServiceDiscovery` by composition and define a custom`findService` function which returns a `std::vector`.
[Check out icediscovery on GitHub :fontawesome-brands-github:](https://github.com/eclipse-iceoryx/iceoryx/tree/v2.0.0/iceoryx_examples/icediscovery){ .md-button }