1 /*
2  * Copyright 2014 Google Inc. All rights reserved.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *     http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef FRUIT_NORMALIZED_COMPONENT_H
18 #define FRUIT_NORMALIZED_COMPONENT_H
19 
20 // This include is not required here, but having it here shortens the include trace in error messages.
21 #include <fruit/impl/injection_errors.h>
22 
23 #include <fruit/fruit_forward_decls.h>
24 #include <fruit/impl/fruit_internal_forward_decls.h>
25 #include <fruit/impl/meta/component.h>
26 #include <fruit/impl/normalized_component_storage/normalized_component_storage_holder.h>
27 #include <memory>
28 
29 namespace fruit {
30 
31 /**
32  * This class allows for fast creation of multiple injectors that share most (or all) the bindings.
33  *
34  * This is an advanced feature of Fruit that allows to reduce injection time in some cases; if you're just starting to
35  * use Fruit you might want to ignore this for now (just construct an Injector from your root Component function).
36  *
37  * Using a NormalizedComponent only helps if:
38  *
39  * - You create multiple injectors during the lifetime of a process. E.g. if you only create one injector at startup you
40  *   won't benefit from using NormalizedComponent.
41  * - Some of those injectors share all (or almost all) their bindings.
42  *
43  * When both of those requirements apply, you can switch to using NormalizedComponent in the "similar" injectors by
44  * first refactoring the injectors' root components to be of the form:
45  *
46  * fruit::Component<...> getRootComponent(...) {
47  *   return fruit::createComponent()
48  *       // This contains the bindings common to the group of similar injectors.
49  *       .install(getSharedComponent, ...)
50  *       // This contains the bindings specific to this injector.
51  *       .install(getSpecificComponent, ...);
52  * }
53  *
54  * Then you can change your injector construction from:
55  *
56  * fruit::Injector<...> injector(getRootComponent, ...);
57  *
58  * To:
59  *
60  * fruit::NormalizedComponent<fruit::Required<...>, ...> normalized_component(getSharedComponent, ...);
61  * fruit::Injector<...> injector(normalized_component, getSpecificComponent, ...);
62  *
63  * This splits the work of constructing the Injector in two phases: normalization (where Fruit will call the Component
64  * functions to collect all the bindings and check for some classes of runtime errors) and the actual creation of the
65  * injector, during which Fruit will also collect/check the additional bindings specific to that injector.
66  *
67  * Then you can share the same normalized_component object across all those injectors (also in different threads,
68  * NormalizedComponent is thread-safe), so that the normalization step only occurs once (i.e., you should only construct
69  * NormalizedComponent from getSharedComponent once, otherwise you'd pay the normalization cost multiple times).
70  *
71  * Creating an Injector from a NormalizedComponent and injecting separate instances is very cheap, on the order of 2 us
72  * for an injection graph with 100 classes and 900 edges (for more details see the Benchmarks page of the Fruit wiki:
73  * https://github.com/google/fruit/wiki/benchmarks ).
74  * This might (depending of course on your performance requirements) allow you to create injectors where it would
75  * otherwise be unthinkable, e.g. creating a separate injector for each request in a server.
76  *
77  * Injectors that share the same NormalizedComponent are still independent; for example, if you call injector.get<Foo>()
78  * in two injectors, each injector will construct its own instance of Foo.
79  *
80  * Example usage in a server:
81  *
82  * // In the global scope.
83  * Component<Request> getRequestComponent(Request* request) {
84  *   return fruit::createComponent()
85  *       .bindInstance(*request);
86  * }
87  *
88  * // At startup (e.g. inside main()).
89  * NormalizedComponent<Required<Request>, Bar, Bar2> normalizedComponent = ...;
90  *
91  * ...
92  * for (...) {
93  *   // For each request.
94  *   Request request = ...;
95  *
96  *   Injector<Foo, Bar> injector(normalizedComponent, getRequestComponent, &request);
97  *   Foo* foo = injector.get<Foo*>();
98  *   ...
99  * }
100  *
101  * See also the documentation for the Injector constructor that takes a NormalizedComponent.
102  */
103 template <typename... Params>
104 class NormalizedComponent {
105 public:
106   /**
107    * The Component used as parameter can have (and usually has) unsatisfied requirements, so it's usually of the form
108    * Component<Required<...>, ...>.
109    *
110    * The given component function is called with the provided arguments to construct the root component.
111    * The constraints on the argument types (if there are any) are the same as the ones for PartialComponent::install().
112    */
113   template <typename... FormalArgs, typename... Args>
114   explicit NormalizedComponent(Component<Params...> (*)(FormalArgs...), Args&&... args);
115 
NormalizedComponent(NormalizedComponent && storage)116   NormalizedComponent(NormalizedComponent&& storage) noexcept : storage(std::move(storage.storage)) {}
117   NormalizedComponent(const NormalizedComponent&) = delete;
118 
119   NormalizedComponent& operator=(NormalizedComponent&&) = delete;
120   NormalizedComponent& operator=(const NormalizedComponent&) = delete;
121 
122 private:
123   NormalizedComponent(fruit::impl::ComponentStorage&& storage, fruit::impl::MemoryPool memory_pool);
124 
125   // This is held via a unique_ptr to avoid including normalized_component_storage.h
126   // in fruit.h.
127   fruit::impl::NormalizedComponentStorageHolder storage;
128 
129   template <typename... OtherParams>
130   friend class Injector;
131 
132   using Comp = fruit::impl::meta::Eval<fruit::impl::meta::ConstructComponentImpl(fruit::impl::meta::Type<Params>...)>;
133 
134   using Check1 = typename fruit::impl::meta::CheckIfError<Comp>::type;
135   // Force instantiation of Check1.
136   static_assert(true || sizeof(Check1), "");
137 };
138 
139 } // namespace fruit
140 
141 #include <fruit/impl/normalized_component.defn.h>
142 
143 #endif // FRUIT_NORMALIZED_COMPONENT_H
144