10. Advanced Mapping - gracicot/kangaru GitHub Wiki

Until now, we only used basic mapping in other examples. The service map was a simple parameter to service mapping. But there's a mechanism to make multiple maps, prioritize them, and use many of them.

The Map Parameter

In a mapping expression, you can actually put a second parameter. That parameter is kgr::map_t<>.

So here's an example of a service map with that parameter:

auto service_map(Service, kgr::map_t<>) -> Definition;

What is it doing? What that parameter for? Well, except for disambiguation for that declaration, not a lot. It will simply make the container prefer this one. Consider this:

auto service_map(Service)               -> Definition1;
auto service_map(Service, kgr::map_t<>) -> Definition2;

When using kgr::mapped_service_t<Service>, that will yield Definition2. Because it has higher priority for the service map.

Named Map

The additional parameter can be feeded with a name. This is how to make a map with a particular name.

A named map in not gonna be used unless to tell the container to use that map.

Every function or metafunction that deal with the service map can have a named map specified. Here are some examples of their usage:

struct MyMap;

auto service_map(Service)                    -> Definition1;
auto service_map(Service, kgr::map_t<MyMap>) -> Definition2;

// Without named parameter, yields Definition1
using UsedDefinition1 = kgr::mapped_service_t<Service>;

// With the named parameter, yields Definition2
using UsedDefinition2 = kgr::mapped_service_t<Service, kgr::map<MyMap>>;

kgr::container container;

auto function = [](Service) {};

container.invoke(function);                  // Definition1 used
container.invoke<kgr::map<MyMap>>(function); // Definition2 used

Also, there's an alternative syntax to send which map is sent to the invoke function:

container.invoke(kgr::map<MyMap>{}, function); // Equivalent

Multiple Maps

The map name parameter we used with other functionalities, the kgr::map<...>, can in fact receive many named map to pick from. The leftmost names will have higher priority. If a map with that name cannot be found, the container will fallback to the next name. If it has no named map left to try, it will try the default map.

Here's an example of multiple maps:

struct MyMap1;
struct MyMap2;

auto service_map(Service1, kgr::map_t<MyMap1>) -> Service1_Definition1;
auto service_map(Service1, kgr::map_t<MyMap2>) -> Service1_Definition2;

auto service_map(Service2, kgr::map_t<MyMap2>) -> Service1_Definition1;
auto service_map(Service2)                     -> Service1_Definition2;

auto function = [](Service1, Service2) {};

container.invoke<kgr::map<MyMap1>>(function); // Service1_Definition1 and Service1_Definition2 used
container.invoke<kgr::map<MyMap2>>(function); // Service1_Definition2 and Service1_Definition2 used
container.invoke<kgr::map<MyMap1, MyMap2>>(function); // Service1_Definition1 and Service1_Definition1 used

Indirect Maps

Since kangaru 4.1.0, the service map allows you de define an indirect mapping. Instead of having the service definition type at the right of the arrow, you can put a class type that has a member template named mapped_service.

It can be used to generate the service definition type automatically by the service map with a decent syntax.

Here's how to create a indirect map:

struct indirect_service {
    template<typename T>
    using mapped_service = kgr::service<std::decay_t<T>> {};
};

auto service_map(Camera const&) -> indirect_service;

if the container found the mapping for camera, instead of trying to use indirect_service as the service type, it will use indirect_service::mapped_service<Camera&>. This in turn will result in kgr::service<Camera>.

It's primary use was for autowiring services, but it can serves many other purposes.

It's also nice to note that when using indirect mapping, the container will try to be stricter in order to avoid generating definitions for unwanted types.

For example, the container won't allow subclasses to be mapped:

struct ServiceBase {};
struct ServiceDerived : ServiceBase {};

struct indirect_service {
    template<typename T>
    using mapped_service = kgr::service<T>;
};

auto service_map(ServiceBase const&) -> misleading_map;

Even though ServiceDerived is convertible to ServiceBase const& and the generated service definition yield the right type, this particular mapping won't be picked, since the mapping was intendended for ServiceBase.

Next chapter: Debugging

⚠️ **GitHub.com Fallback** ⚠️