#!/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()