mirror of
				https://github.com/ets-labs/python-dependency-injector.git
				synced 2025-11-04 01:47:36 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			160 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			160 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
"""Dependency injector dynamic container unit tests for async resources."""
 | 
						|
import asyncio
 | 
						|
 | 
						|
# Runtime import to get asyncutils module
 | 
						|
import os
 | 
						|
_TOP_DIR = os.path.abspath(
 | 
						|
    os.path.sep.join((
 | 
						|
        os.path.dirname(__file__),
 | 
						|
        "../",
 | 
						|
    )),
 | 
						|
)
 | 
						|
import sys
 | 
						|
sys.path.append(_TOP_DIR)
 | 
						|
 | 
						|
from asyncutils import AsyncTestCase
 | 
						|
 | 
						|
from dependency_injector import (
 | 
						|
    containers,
 | 
						|
    providers,
 | 
						|
)
 | 
						|
 | 
						|
 | 
						|
class AsyncResourcesTest(AsyncTestCase):
 | 
						|
 | 
						|
    def test_init_and_shutdown_ordering(self):
 | 
						|
        """Test init and shutdown resources.
 | 
						|
 | 
						|
        Methods .init_resources() and .shutdown_resources() should respect resources dependencies.
 | 
						|
        Initialization should first initialize resources without dependencies and then provide
 | 
						|
        these resources to other resources. Resources shutdown should follow the same rule: first
 | 
						|
        shutdown resources without initialized dependencies and then continue correspondingly
 | 
						|
        until all resources are shutdown.
 | 
						|
        """
 | 
						|
        initialized_resources = []
 | 
						|
        shutdown_resources = []
 | 
						|
 | 
						|
        async def _resource(name, delay, **_):
 | 
						|
            await asyncio.sleep(delay)
 | 
						|
            initialized_resources.append(name)
 | 
						|
 | 
						|
            yield name
 | 
						|
 | 
						|
            await asyncio.sleep(delay)
 | 
						|
            shutdown_resources.append(name)
 | 
						|
 | 
						|
        class Container(containers.DeclarativeContainer):
 | 
						|
            resource1 = providers.Resource(
 | 
						|
                _resource,
 | 
						|
                name="r1",
 | 
						|
                delay=0.03,
 | 
						|
            )
 | 
						|
            resource2 = providers.Resource(
 | 
						|
                _resource,
 | 
						|
                name="r2",
 | 
						|
                delay=0.02,
 | 
						|
                r1=resource1,
 | 
						|
            )
 | 
						|
            resource3 = providers.Resource(
 | 
						|
                _resource,
 | 
						|
                name="r3",
 | 
						|
                delay=0.01,
 | 
						|
                r2=resource2,
 | 
						|
            )
 | 
						|
 | 
						|
        container = Container()
 | 
						|
 | 
						|
        self._run(container.init_resources())
 | 
						|
        self.assertEqual(initialized_resources, ["r1", "r2", "r3"])
 | 
						|
        self.assertEqual(shutdown_resources, [])
 | 
						|
 | 
						|
        self._run(container.shutdown_resources())
 | 
						|
        self.assertEqual(initialized_resources, ["r1", "r2", "r3"])
 | 
						|
        self.assertEqual(shutdown_resources, ["r3", "r2", "r1"])
 | 
						|
 | 
						|
        self._run(container.init_resources())
 | 
						|
        self.assertEqual(initialized_resources, ["r1", "r2", "r3", "r1", "r2", "r3"])
 | 
						|
        self.assertEqual(shutdown_resources, ["r3", "r2", "r1"])
 | 
						|
 | 
						|
        self._run(container.shutdown_resources())
 | 
						|
        self.assertEqual(initialized_resources, ["r1", "r2", "r3", "r1", "r2", "r3"])
 | 
						|
        self.assertEqual(shutdown_resources, ["r3", "r2", "r1", "r3", "r2", "r1"])
 | 
						|
 | 
						|
    def test_shutdown_circular_dependencies_breaker(self):
 | 
						|
        async def _resource(name, **_):
 | 
						|
            yield name
 | 
						|
 | 
						|
        class Container(containers.DeclarativeContainer):
 | 
						|
            resource1 = providers.Resource(
 | 
						|
                _resource,
 | 
						|
                name="r1",
 | 
						|
            )
 | 
						|
            resource2 = providers.Resource(
 | 
						|
                _resource,
 | 
						|
                name="r2",
 | 
						|
                r1=resource1,
 | 
						|
            )
 | 
						|
            resource3 = providers.Resource(
 | 
						|
                _resource,
 | 
						|
                name="r3",
 | 
						|
                r2=resource2,
 | 
						|
            )
 | 
						|
 | 
						|
        container = Container()
 | 
						|
        self._run(container.init_resources())
 | 
						|
 | 
						|
        # Create circular dependency after initialization (r3 -> r2 -> r1 -> r3 -> ...)
 | 
						|
        container.resource1.add_kwargs(r3=container.resource3)
 | 
						|
 | 
						|
        with self.assertRaises(RuntimeError) as context:
 | 
						|
            self._run(container.shutdown_resources())
 | 
						|
        self.assertEqual(str(context.exception), "Unable to resolve resources shutdown order")
 | 
						|
 | 
						|
    def test_shutdown_sync_and_async_ordering(self):
 | 
						|
        initialized_resources = []
 | 
						|
        shutdown_resources = []
 | 
						|
 | 
						|
        def _sync_resource(name, **_):
 | 
						|
            initialized_resources.append(name)
 | 
						|
            yield name
 | 
						|
            shutdown_resources.append(name)
 | 
						|
 | 
						|
        async def _async_resource(name, **_):
 | 
						|
            initialized_resources.append(name)
 | 
						|
            yield name
 | 
						|
            shutdown_resources.append(name)
 | 
						|
 | 
						|
        class Container(containers.DeclarativeContainer):
 | 
						|
            resource1 = providers.Resource(
 | 
						|
                _sync_resource,
 | 
						|
                name="r1",
 | 
						|
            )
 | 
						|
            resource2 = providers.Resource(
 | 
						|
                _sync_resource,
 | 
						|
                name="r2",
 | 
						|
                r1=resource1,
 | 
						|
            )
 | 
						|
            resource3 = providers.Resource(
 | 
						|
                _async_resource,
 | 
						|
                name="r3",
 | 
						|
                r2=resource2,
 | 
						|
            )
 | 
						|
 | 
						|
        container = Container()
 | 
						|
 | 
						|
        self._run(container.init_resources())
 | 
						|
        self.assertEqual(initialized_resources, ["r1", "r2", "r3"])
 | 
						|
        self.assertEqual(shutdown_resources, [])
 | 
						|
 | 
						|
        self._run(container.shutdown_resources())
 | 
						|
        self.assertEqual(initialized_resources, ["r1", "r2", "r3"])
 | 
						|
        self.assertEqual(shutdown_resources, ["r3", "r2", "r1"])
 | 
						|
 | 
						|
        self._run(container.init_resources())
 | 
						|
        self.assertEqual(initialized_resources, ["r1", "r2", "r3", "r1", "r2", "r3"])
 | 
						|
        self.assertEqual(shutdown_resources, ["r3", "r2", "r1"])
 | 
						|
 | 
						|
        self._run(container.shutdown_resources())
 | 
						|
        self.assertEqual(initialized_resources, ["r1", "r2", "r3", "r1", "r2", "r3"])
 | 
						|
        self.assertEqual(shutdown_resources, ["r3", "r2", "r1", "r3", "r2", "r1"])
 |