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.
210 lines
7.3 KiB
210 lines
7.3 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.
|
|
import pytest
|
|
from absl.testing import parameterized
|
|
from fruit_test_common import *
|
|
from fruit_test_config import CXX_COMPILER_NAME
|
|
import re
|
|
|
|
COMMON_DEFINITIONS = '''
|
|
#include "test_common.h"
|
|
|
|
struct Annotation {};
|
|
struct Annotation1 {};
|
|
struct Annotation2 {};
|
|
'''
|
|
|
|
class TestMultibindingsBindInterface(parameterized.TestCase):
|
|
@parameterized.parameters([
|
|
('X', 'XImpl'),
|
|
('X', 'fruit::Annotated<Annotation2, XImpl>'),
|
|
('fruit::Annotated<Annotation1, X>', 'XImpl'),
|
|
('fruit::Annotated<Annotation1, X>', 'fruit::Annotated<Annotation2, XImpl>'),
|
|
])
|
|
def test_add_interface_multibinding_success(self, XAnnot, XImplAnnot):
|
|
source = '''
|
|
struct X {
|
|
virtual int foo() = 0;
|
|
};
|
|
|
|
struct XImpl : public X {
|
|
INJECT(XImpl()) = default;
|
|
|
|
int foo() override {
|
|
return 5;
|
|
}
|
|
};
|
|
|
|
fruit::Component<> getComponent() {
|
|
return fruit::createComponent()
|
|
.addMultibinding<XAnnot, XImplAnnot>();
|
|
}
|
|
|
|
int main() {
|
|
fruit::Injector<> injector(getComponent);
|
|
|
|
std::vector<X*> multibindings = injector.getMultibindings<XAnnot>();
|
|
Assert(multibindings.size() == 1);
|
|
Assert(multibindings[0]->foo() == 5);
|
|
}
|
|
'''
|
|
expect_success(
|
|
COMMON_DEFINITIONS,
|
|
source,
|
|
locals())
|
|
|
|
@parameterized.parameters([
|
|
('X', 'XImpl', 'const XImpl'),
|
|
('X', 'fruit::Annotated<Annotation2, XImpl>', 'fruit::Annotated<Annotation2, const XImpl>'),
|
|
])
|
|
def test_add_interface_multibinding_const_target_error_install_first(self, XAnnot, XImplAnnot, ConstXImplAnnot):
|
|
source = '''
|
|
struct X {
|
|
virtual int foo() = 0;
|
|
};
|
|
|
|
struct XImpl : public X {
|
|
int foo() override {
|
|
return 5;
|
|
}
|
|
};
|
|
|
|
fruit::Component<ConstXImplAnnot> getXImplComponent();
|
|
|
|
fruit::Component<> getComponent() {
|
|
return fruit::createComponent()
|
|
.install(getXImplComponent)
|
|
.addMultibinding<XAnnot, XImplAnnot>();
|
|
}
|
|
'''
|
|
expect_compile_error(
|
|
'NonConstBindingRequiredButConstBindingProvidedError<XImplAnnot>',
|
|
'The type T was provided as constant, however one of the constructors/providers/factories in this component',
|
|
COMMON_DEFINITIONS,
|
|
source,
|
|
locals())
|
|
|
|
@parameterized.parameters([
|
|
('X', 'XImpl', 'const XImpl'),
|
|
('X', 'fruit::Annotated<Annotation2, XImpl>', 'fruit::Annotated<Annotation2, const XImpl>'),
|
|
])
|
|
def test_add_interface_multibinding_const_target_error_binding_first(self, XAnnot, XImplAnnot, ConstXImplAnnot):
|
|
source = '''
|
|
struct X {
|
|
virtual int foo() = 0;
|
|
};
|
|
|
|
struct XImpl : public X {
|
|
int foo() override {
|
|
return 5;
|
|
}
|
|
};
|
|
|
|
fruit::Component<ConstXImplAnnot> getXImplComponent();
|
|
|
|
fruit::Component<> getComponent() {
|
|
return fruit::createComponent()
|
|
.addMultibinding<XAnnot, XImplAnnot>()
|
|
.install(getXImplComponent);
|
|
}
|
|
'''
|
|
expect_compile_error(
|
|
'NonConstBindingRequiredButConstBindingProvidedError<XImplAnnot>',
|
|
'The type T was provided as constant, however one of the constructors/providers/factories in this component',
|
|
COMMON_DEFINITIONS,
|
|
source,
|
|
locals())
|
|
|
|
@parameterized.parameters([
|
|
('X', 'int'),
|
|
('fruit::Annotated<Annotation1, X>', 'fruit::Annotated<Annotation2, int>'),
|
|
])
|
|
def test_error_not_base(self, XAnnot, intAnnot):
|
|
source = '''
|
|
struct X {};
|
|
|
|
fruit::Component<> getComponent() {
|
|
return fruit::createComponent()
|
|
.addMultibinding<XAnnot, intAnnot>();
|
|
}
|
|
'''
|
|
expect_compile_error(
|
|
'NotABaseClassOfError<X,int>',
|
|
'I is not a base class of C.',
|
|
COMMON_DEFINITIONS,
|
|
source,
|
|
locals())
|
|
|
|
@parameterized.parameters([
|
|
('Scaler', 'ScalerImpl'),
|
|
('fruit::Annotated<Annotation1, Scaler>', 'fruit::Annotated<Annotation2, ScalerImpl>'),
|
|
])
|
|
def test_error_abstract_class(self, ScalerAnnot, ScalerImplAnnot):
|
|
source = '''
|
|
struct Scaler {
|
|
virtual double scale(double x) = 0;
|
|
};
|
|
|
|
struct ScalerImpl : public Scaler {
|
|
// Note: here we "forgot" to implement scale() (on purpose, for this test) so ScalerImpl is an abstract class.
|
|
};
|
|
|
|
fruit::Component<> getComponent() {
|
|
return fruit::createComponent()
|
|
.addMultibinding<ScalerAnnot, ScalerImplAnnot>();
|
|
}
|
|
'''
|
|
expect_compile_error(
|
|
'NoBindingFoundForAbstractClassError<ScalerImplAnnot,ScalerImpl>',
|
|
'No explicit binding was found for T, and note that C is an abstract class',
|
|
COMMON_DEFINITIONS,
|
|
source,
|
|
locals())
|
|
|
|
@parameterized.parameters([
|
|
('Scaler', 'ScalerImpl'),
|
|
('fruit::Annotated<Annotation1, Scaler>', 'fruit::Annotated<Annotation2, ScalerImpl>'),
|
|
])
|
|
def test_error_abstract_class_clang(self, ScalerAnnot, ScalerImplAnnot):
|
|
if re.search('Clang', CXX_COMPILER_NAME) is None:
|
|
# This is Clang-only because GCC >=4.9 refuses to even mention the type C() when C is an abstract class,
|
|
# while Clang allows to mention the type (but of course there can be no functions with this type)
|
|
return
|
|
source = '''
|
|
struct Scaler {
|
|
virtual double scale(double x) = 0;
|
|
};
|
|
|
|
struct ScalerImpl : public Scaler {
|
|
INJECT(ScalerImpl()) = default;
|
|
|
|
// Note: here we "forgot" to implement scale() (on purpose, for this test) so ScalerImpl is an abstract class.
|
|
};
|
|
|
|
fruit::Component<> getComponent() {
|
|
return fruit::createComponent()
|
|
.addMultibinding<ScalerAnnot, ScalerImplAnnot>();
|
|
}
|
|
'''
|
|
expect_compile_error(
|
|
'CannotConstructAbstractClassError<ScalerImpl>',
|
|
'The specified class can.t be constructed because it.s an abstract class.',
|
|
COMMON_DEFINITIONS,
|
|
source,
|
|
locals())
|
|
|
|
if __name__ == '__main__':
|
|
absltest.main()
|