You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
163 lines
6.0 KiB
163 lines
6.0 KiB
#!/usr/bin/env python3
|
|
# Copyright 2016 Google Inc. All Rights Reserved.
|
|
#
|
|
# 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.
|
|
|
|
from absl.testing import parameterized
|
|
from fruit_test_common import *
|
|
|
|
COMMON_DEFINITIONS = '''
|
|
#include "test_common.h"
|
|
|
|
struct Annotation1 {};
|
|
struct Annotation2 {};
|
|
|
|
template <typename T>
|
|
using WithNoAnnot = T;
|
|
|
|
template <typename T>
|
|
using WithAnnot1 = fruit::Annotated<Annotation1, T>;
|
|
|
|
template <typename T>
|
|
using WithAnnot2 = fruit::Annotated<Annotation2, T>;
|
|
'''
|
|
|
|
class TestBindingCompression(parameterized.TestCase):
|
|
@parameterized.parameters([
|
|
('I', 'X', 'WithNoAnnot'),
|
|
('fruit::Annotated<Annotation1, I>', 'fruit::Annotated<Annotation2, X>', 'WithAnnot1'),
|
|
])
|
|
def test_provider_returning_value_success_with_annotation(self, IAnnot, XAnnot, WithAnnot):
|
|
source = '''
|
|
struct I {
|
|
int value = 5;
|
|
};
|
|
|
|
struct X : public I, ConstructionTracker<X> {
|
|
};
|
|
|
|
fruit::Component<IAnnot> getComponent() {
|
|
return fruit::createComponent()
|
|
.registerProvider<XAnnot()>([](){return X();})
|
|
.bind<IAnnot, XAnnot>();
|
|
}
|
|
|
|
int main() {
|
|
fruit::Injector<IAnnot> injector(getComponent);
|
|
Assert((injector.get<WithAnnot<I >>() .value == 5));
|
|
Assert((injector.get<WithAnnot<I* >>()->value == 5));
|
|
Assert((injector.get<WithAnnot<I& >>() .value == 5));
|
|
Assert((injector.get<WithAnnot<const I >>() .value == 5));
|
|
Assert((injector.get<WithAnnot<const I* >>()->value == 5));
|
|
Assert((injector.get<WithAnnot<const I& >>() .value == 5));
|
|
Assert((injector.get<WithAnnot<std::shared_ptr<I>>>()->value == 5));
|
|
Assert(fruit::impl::InjectorAccessorForTests::unsafeGet<WithAnnot<X>>(injector) == nullptr);
|
|
|
|
Assert(X::num_objects_constructed == 1);
|
|
}
|
|
'''
|
|
expect_success(
|
|
COMMON_DEFINITIONS,
|
|
source,
|
|
locals())
|
|
|
|
@parameterized.parameters([
|
|
('I', 'X', 'X*', 'WithNoAnnot'),
|
|
('fruit::Annotated<Annotation1, I>', 'fruit::Annotated<Annotation2, X>', 'fruit::Annotated<Annotation2, X*>', 'WithAnnot1'),
|
|
])
|
|
def test_provider_returning_pointer_success_with_annotation(self, IAnnot, XAnnot, XPtrAnnot, WithAnnot):
|
|
source = '''
|
|
struct I {
|
|
int value = 5;
|
|
};
|
|
|
|
struct X : public I, ConstructionTracker<X> {
|
|
};
|
|
|
|
fruit::Component<IAnnot> getComponent() {
|
|
return fruit::createComponent()
|
|
.registerProvider<XPtrAnnot()>([](){return new X();})
|
|
.bind<IAnnot, XAnnot>();
|
|
}
|
|
|
|
int main() {
|
|
fruit::Injector<IAnnot> injector(getComponent);
|
|
Assert((injector.get<WithAnnot<I >>() .value == 5));
|
|
Assert((injector.get<WithAnnot<I* >>()->value == 5));
|
|
Assert((injector.get<WithAnnot<I& >>() .value == 5));
|
|
Assert((injector.get<WithAnnot<const I >>() .value == 5));
|
|
Assert((injector.get<WithAnnot<const I* >>()->value == 5));
|
|
Assert((injector.get<WithAnnot<const I& >>() .value == 5));
|
|
Assert((injector.get<WithAnnot<std::shared_ptr<I>>>()->value == 5));
|
|
Assert(fruit::impl::InjectorAccessorForTests::unsafeGet<WithAnnot<X>>(injector) == nullptr);
|
|
Assert(X::num_objects_constructed == 1);
|
|
}
|
|
'''
|
|
expect_success(
|
|
COMMON_DEFINITIONS,
|
|
source,
|
|
locals())
|
|
|
|
def test_compression_undone(self):
|
|
source = '''
|
|
struct I1 {};
|
|
struct C1 : public I1, ConstructionTracker<C1> {
|
|
INJECT(C1()) = default;
|
|
};
|
|
|
|
struct I2 {};
|
|
struct C2 : public I2 {
|
|
INJECT(C2(I1*)) {}
|
|
};
|
|
|
|
fruit::Component<I1> getI1Component() {
|
|
return fruit::createComponent()
|
|
.bind<I1, C1>();
|
|
}
|
|
|
|
fruit::Component<I2> getI2Component() {
|
|
return fruit::createComponent()
|
|
.install(getI1Component)
|
|
.bind<I2, C2>();
|
|
}
|
|
|
|
struct X {
|
|
// Intentionally C1 and not I1. This prevents binding compression for the I1->C1 edge.
|
|
INJECT(X(C1*)) {}
|
|
};
|
|
|
|
fruit::Component<X> getXComponent() {
|
|
return fruit::createComponent();
|
|
}
|
|
|
|
int main() {
|
|
// Here the binding C2->I1->C1 is compressed into C2->C1.
|
|
fruit::NormalizedComponent<I2> normalizedComponent(getI2Component);
|
|
|
|
// However the binding X->C1 prevents binding compression on I1->C1, the binding compression must be undone.
|
|
fruit::Injector<I2, X> injector(normalizedComponent, getXComponent);
|
|
|
|
Assert(C1::num_objects_constructed == 0);
|
|
injector.get<I2*>();
|
|
injector.get<X*>();
|
|
Assert(fruit::impl::InjectorAccessorForTests::unsafeGet<C1>(injector) != nullptr);
|
|
Assert(C1::num_objects_constructed == 1);
|
|
}
|
|
'''
|
|
expect_success(
|
|
COMMON_DEFINITIONS,
|
|
source)
|
|
|
|
if __name__ == '__main__':
|
|
absltest.main()
|