#!/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 X; struct Annotation1 {}; using XAnnot1 = fruit::Annotated; struct Annotation2 {}; using XAnnot2 = fruit::Annotated; struct Annotation3 {}; using XAnnot3 = fruit::Annotated; ''' @pytest.mark.parametrize('XAnnot,XConstRefAnnot,YAnnot,YConstRefAnnot', [ ('X', 'const X&', 'Y', 'const Y&'), ('fruit::Annotated', 'ANNOTATED(Annotation1, const X&)', 'fruit::Annotated', 'ANNOTATED(Annotation2, const Y&)') ]) def test_loop_in_autoinject(XAnnot, XConstRefAnnot, YAnnot, YConstRefAnnot): source = ''' struct Y; struct X { INJECT(X(YConstRefAnnot)) {}; }; struct Y { INJECT(Y(XConstRefAnnot)) {}; }; fruit::Component mutuallyConstructibleComponent() { return fruit::createComponent(); } ''' expect_compile_error( 'SelfLoopError', 'Found a loop in the dependencies', COMMON_DEFINITIONS, source, locals()) @pytest.mark.parametrize('XAnnot,ConstXAnnot,XConstRefAnnot,YAnnot,YConstRefAnnot', [ ('X', 'const X', 'const X&', 'Y', 'const Y&'), ('fruit::Annotated', 'ANNOTATED(Annotation1, const X)', 'ANNOTATED(Annotation1, const X&)', 'fruit::Annotated', 'ANNOTATED(Annotation2, const Y&)') ]) def test_loop_in_autoinject_const(XAnnot, ConstXAnnot, XConstRefAnnot, YAnnot, YConstRefAnnot): source = ''' struct Y; struct X { INJECT(X(YConstRefAnnot)) {}; }; struct Y { INJECT(Y(XConstRefAnnot)) {}; }; fruit::Component mutuallyConstructibleComponent() { return fruit::createComponent(); } ''' expect_compile_error( 'SelfLoopError', 'Found a loop in the dependencies', COMMON_DEFINITIONS, source, locals()) def test_loop_in_register_provider(): source = ''' struct X {}; struct Y {}; fruit::Component mutuallyConstructibleComponent() { return fruit::createComponent() .registerProvider([](Y) {return X();}) .registerProvider([](X) {return Y();}); } ''' expect_compile_error( 'SelfLoopError', 'Found a loop in the dependencies', COMMON_DEFINITIONS, source, locals()) def test_loop_in_register_provider_with_annotations(): source = ''' struct X {}; fruit::Component> mutuallyConstructibleComponent() { return fruit::createComponent() .registerProvider(fruit::Annotated)>([](X x) {return x;}) .registerProvider(fruit::Annotated)>([](X x) {return x;}); } ''' expect_compile_error( 'SelfLoopError, fruit::Annotated>', 'Found a loop in the dependencies', COMMON_DEFINITIONS, source, locals()) def test_with_different_annotations_ok(): source = ''' struct X {}; fruit::Component getComponent() { return fruit::createComponent() .registerProvider([](){return X();}) .registerProvider([](X x){return x;}) .registerProvider([](X x){return x;}); } int main() { fruit::Injector injector(getComponent); injector.get(); } ''' expect_success( COMMON_DEFINITIONS, source) if __name__== '__main__': main(__file__)