#!/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 fruit_test_common import *
COMMON_DEFINITIONS = '''
#include "test_common.h"
struct Annotation1 {};
'''
@pytest.mark.parametrize('XAnnot', [
'X',
'fruit::Annotated<Annotation1, X>',
])
def test_multibindings_bind_instance_ok(XAnnot):
source = '''
struct X {};
X x;
fruit::Component<> getComponent() {
return fruit::createComponent()
.addInstanceMultibinding<XAnnot, X>(x);
}
int main() {
fruit::Injector<> injector(getComponent);
std::vector<X*> multibindings = injector.getMultibindings<XAnnot>();
Assert(multibindings.size() == 1);
Assert(multibindings[0] == &x);
}
'''
expect_success(
COMMON_DEFINITIONS,
source,
locals())
@pytest.mark.parametrize('XAnnot', [
'X',
'fruit::Annotated<Annotation1, X>',
])
def test_multibindings_bind_const_instance_error(XAnnot):
source = '''
struct X {};
const X x{};
fruit::Component<> getComponent() {
return fruit::createComponent()
.addInstanceMultibinding<XAnnot, X>(x);
}
'''
expect_generic_compile_error(
'candidate function not viable: 1st argument \(.const X.\) would lose const qualifier'
'|no matching function for call to .fruit::PartialComponent<.*>::addInstanceMultibinding(<XAnnot,X>)?\(const X&\).'
'|error: no matching member function for call to .addInstanceMultibinding.'
'|cannot convert argument 1 from .const X. to .X &.',
COMMON_DEFINITIONS,
source,
locals())
@pytest.mark.parametrize('XAnnot', [
'X',
'fruit::Annotated<Annotation1, X>',
])
def test_multibindings_bind_instance_vector(XAnnot):
source = '''
struct X {};
std::vector<X> values = {X(), X()};
fruit::Component<> getComponent() {
return fruit::createComponent()
.addInstanceMultibindings<XAnnot, X>(values);
}
int main() {
fruit::Injector<> injector(getComponent);
std::vector<X*> multibindings = injector.getMultibindings<XAnnot>();
Assert(multibindings.size() == 2);
Assert(multibindings[0] == &(values[0]));
Assert(multibindings[1] == &(values[1]));
}
'''
expect_success(
COMMON_DEFINITIONS,
source,
locals())
@pytest.mark.parametrize('XAnnot', [
'X',
'fruit::Annotated<Annotation1, X>',
])
def test_multibindings_bind_const_instance_vector_error(XAnnot):
source = '''
struct X {};
const std::vector<X> values{};
fruit::Component<> getComponent() {
return fruit::createComponent()
.addInstanceMultibindings<XAnnot, X>(values);
}
'''
expect_generic_compile_error(
'candidate function not viable: 1st argument \(.const std::vector<X>.\) would lose const qualifier'
'|cannot convert .values. \(type .const std::(__debug::)?vector<X>.\) to type .std::(__debug::)?vector<X>&.'
'|no matching member function for call to .addInstanceMultibindings.'
'|cannot convert argument 1 from .const std::vector<X,std::allocator<.*>>. to .std::vector<X,std::allocator<.*>> &.',
COMMON_DEFINITIONS,
source,
locals())
@pytest.mark.parametrize('XAnnot', [
'X',
'fruit::Annotated<Annotation1, X>',
])
def test_multibindings_bind_instance_vector_of_consts_error(XAnnot):
source = '''
struct X {};
std::vector<const X> values;
fruit::Component<> getComponent() {
return fruit::createComponent()
.addInstanceMultibindings<XAnnot, X>(values);
}
'''
expect_generic_compile_error(
'.*',
COMMON_DEFINITIONS,
source,
locals())
@pytest.mark.parametrize('XVariant,XVariantRegex', [
('X**', r'X\*\*'),
('std::shared_ptr<X>*', r'std::shared_ptr<X>\*'),
('const std::shared_ptr<X>', r'const std::shared_ptr<X>'),
('X* const', r'X\* const'),
('const X* const', r'const X\* const'),
('X*&', r'X\*&'),
('fruit::Annotated<Annotation1, X**>', r'X\*\*'),
])
def test_multibindings_bind_instance_non_class_type_error(XVariant, XVariantRegex):
source = '''
struct X {};
using XVariantT = XVariant;
fruit::Component<> getComponent(XVariantT x) {
return fruit::createComponent()
.addInstanceMultibinding<XVariant, XVariant>(x);
}
'''
expect_compile_error(
'NonClassTypeError<XVariantRegex,X>',
'A non-class type T was specified.',
COMMON_DEFINITIONS,
source,
locals())
@pytest.mark.parametrize('XVariant,XVariantRegex', [
('std::nullptr_t', r'(std::)?nullptr(_t)?'),
('X(*)()', r'X(\((__cdecl)?\*\))?\((void)?\)'),
])
def test_multibindings_bind_instance_non_injectable_type_error(XVariant, XVariantRegex):
source = '''
struct X {};
using XVariantT = XVariant;
fruit::Component<> getComponent(XVariantT x) {
return fruit::createComponent()
.addInstanceMultibinding<XVariant, XVariant>(x);
}
'''
expect_compile_error(
'NonInjectableTypeError<XVariantRegex>',
'The type T is not injectable.',
COMMON_DEFINITIONS,
source,
locals())
if __name__== '__main__':
main(__file__)