import unittest.mock import importlib from typing import Any utils = importlib.import_module('extensions.sd-webui-controlnet.tests.utils', 'utils') utils.setup_test_env() from modules import processing, scripts, shared from scripts import controlnet, external_code, batch_hijack batch_hijack.instance.undo_hijack() original_process_images_inner = processing.process_images_inner class TestBatchHijack(unittest.TestCase): @unittest.mock.patch('modules.script_callbacks.on_script_unloaded') def setUp(self, on_script_unloaded_mock): self.on_script_unloaded_mock = on_script_unloaded_mock self.batch_hijack_object = batch_hijack.BatchHijack() self.batch_hijack_object.do_hijack() def tearDown(self): self.batch_hijack_object.undo_hijack() def test_do_hijack__registers_on_script_unloaded(self): self.on_script_unloaded_mock.assert_called_once_with(self.batch_hijack_object.undo_hijack) def test_do_hijack__call_once__hijacks_once(self): self.assertEqual(getattr(processing, '__controlnet_original_process_images_inner'), original_process_images_inner) self.assertEqual(processing.process_images_inner, self.batch_hijack_object.processing_process_images_hijack) @unittest.mock.patch('modules.processing.__controlnet_original_process_images_inner') def test_do_hijack__multiple_times__hijacks_once(self, process_images_inner_mock): self.batch_hijack_object.do_hijack() self.batch_hijack_object.do_hijack() self.batch_hijack_object.do_hijack() self.assertEqual(process_images_inner_mock, getattr(processing, '__controlnet_original_process_images_inner')) class TestGetControlNetBatchesWorks(unittest.TestCase): def setUp(self): self.p = unittest.mock.MagicMock() self.p.scripts = scripts.scripts_txt2img self.cn_script = controlnet.Script() self.p.scripts.alwayson_scripts = [self.cn_script] self.p.script_args = [] def tearDown(self): batch_hijack.instance.dispatch_callbacks(batch_hijack.instance.postprocess_batch_callbacks, self.p) def assert_get_cn_batches_works(self, batch_images_list): self.cn_script.args_from = 0 self.cn_script.args_to = self.cn_script.args_from + len(self.p.script_args) is_cn_batch, batches, output_dir, _ = batch_hijack.get_cn_batches(self.p) batch_hijack.instance.dispatch_callbacks(batch_hijack.instance.process_batch_callbacks, self.p, batches, output_dir) batch_units = [unit for unit in self.p.script_args if getattr(unit, 'input_mode', batch_hijack.InputMode.SIMPLE) == batch_hijack.InputMode.BATCH] if batch_units: self.assertEqual(min(len(unit.batch_images) for unit in batch_units), len(batches)) else: self.assertEqual(1, len(batches)) for i, unit in enumerate(self.cn_script.enabled_units): self.assertListEqual(batch_images_list[i], list(unit.batch_images)) def test_get_cn_batches__empty(self): is_batch, batches, _, _ = batch_hijack.get_cn_batches(self.p) self.assertEqual(1, len(batches)) self.assertEqual(is_batch, False) def test_get_cn_batches__1_simple(self): self.p.script_args.append(external_code.ControlNetUnit(image=get_dummy_image())) self.assert_get_cn_batches_works([ [self.p.script_args[0].image], ]) def test_get_cn_batches__2_simples(self): self.p.script_args.extend([ external_code.ControlNetUnit(image=get_dummy_image(0)), external_code.ControlNetUnit(image=get_dummy_image(1)), ]) self.assert_get_cn_batches_works([ [get_dummy_image(0)], [get_dummy_image(1)], ]) def test_get_cn_batches__1_batch(self): self.p.script_args.extend([ controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[ get_dummy_image(0), get_dummy_image(1), ], ), ]) self.assert_get_cn_batches_works([ [ get_dummy_image(0), get_dummy_image(1), ], ]) def test_get_cn_batches__2_batches(self): self.p.script_args.extend([ controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[ get_dummy_image(0), get_dummy_image(1), ], ), controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[ get_dummy_image(2), get_dummy_image(3), ], ), ]) self.assert_get_cn_batches_works([ [ get_dummy_image(0), get_dummy_image(1), ], [ get_dummy_image(2), get_dummy_image(3), ], ]) def test_get_cn_batches__2_mixed(self): self.p.script_args.extend([ external_code.ControlNetUnit(image=get_dummy_image(0)), controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[ get_dummy_image(1), get_dummy_image(2), ], ), ]) self.assert_get_cn_batches_works([ [ get_dummy_image(0), get_dummy_image(0), ], [ get_dummy_image(1), get_dummy_image(2), ], ]) def test_get_cn_batches__3_mixed(self): self.p.script_args.extend([ external_code.ControlNetUnit(image=get_dummy_image(0)), controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[ get_dummy_image(1), get_dummy_image(2), get_dummy_image(3), ], ), controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[ get_dummy_image(4), get_dummy_image(5), ], ), ]) self.assert_get_cn_batches_works([ [ get_dummy_image(0), get_dummy_image(0), ], [ get_dummy_image(1), get_dummy_image(2), ], [ get_dummy_image(4), get_dummy_image(5), ], ]) class TestProcessImagesPatchWorks(unittest.TestCase): @unittest.mock.patch('modules.script_callbacks.on_script_unloaded') def setUp(self, on_script_unloaded_mock): self.on_script_unloaded_mock = on_script_unloaded_mock self.p = unittest.mock.MagicMock() self.p.scripts = scripts.scripts_txt2img self.cn_script = controlnet.Script() self.p.scripts.alwayson_scripts = [self.cn_script] self.p.script_args = [] self.p.all_seeds = [0] self.p.all_subseeds = [0] self.old_model, shared.sd_model = shared.sd_model, unittest.mock.MagicMock() self.batch_hijack_object = batch_hijack.BatchHijack() self.callbacks_mock = unittest.mock.MagicMock() self.batch_hijack_object.process_batch_callbacks.append(self.callbacks_mock.process) self.batch_hijack_object.process_batch_each_callbacks.append(self.callbacks_mock.process_each) self.batch_hijack_object.postprocess_batch_each_callbacks.insert(0, self.callbacks_mock.postprocess_each) self.batch_hijack_object.postprocess_batch_callbacks.insert(0, self.callbacks_mock.postprocess) self.batch_hijack_object.do_hijack() shared.state.begin() def tearDown(self): shared.state.end() self.batch_hijack_object.undo_hijack() shared.sd_model = self.old_model @unittest.mock.patch('modules.processing.__controlnet_original_process_images_inner') def assert_process_images_hijack_called(self, process_images_mock, batch_count): process_images_mock.return_value = processing.Processed(self.p, [get_dummy_image('output')]) with unittest.mock.patch.dict(shared.opts.data, { 'controlnet_show_batch_images_in_ui': True, }): res = processing.process_images_inner(self.p) self.assertEqual(res, process_images_mock.return_value) if batch_count > 0: self.callbacks_mock.process.assert_called() self.callbacks_mock.postprocess.assert_called() else: self.callbacks_mock.process.assert_not_called() self.callbacks_mock.postprocess.assert_not_called() self.assertEqual(self.callbacks_mock.process_each.call_count, batch_count) self.assertEqual(self.callbacks_mock.postprocess_each.call_count, batch_count) def test_process_images_no_units_forwards(self): self.assert_process_images_hijack_called(batch_count=0) def test_process_images__only_simple_units__forwards(self): self.p.script_args = [ external_code.ControlNetUnit(image=get_dummy_image()), external_code.ControlNetUnit(image=get_dummy_image()), ] self.assert_process_images_hijack_called(batch_count=0) def test_process_images__1_batch_1_unit__runs_1_batch(self): self.p.script_args = [ controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[ get_dummy_image(), ], ), ] self.assert_process_images_hijack_called(batch_count=1) def test_process_images__2_batches_1_unit__runs_2_batches(self): self.p.script_args = [ controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[ get_dummy_image(0), get_dummy_image(1), ], ), ] self.assert_process_images_hijack_called(batch_count=2) def test_process_images__8_batches_1_unit__runs_8_batches(self): batch_count = 8 self.p.script_args = [ controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[get_dummy_image(i) for i in range(batch_count)] ), ] self.assert_process_images_hijack_called(batch_count=batch_count) def test_process_images__1_batch_2_units__runs_1_batch(self): self.p.script_args = [ controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[get_dummy_image(0)] ), controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[get_dummy_image(1)] ), ] self.assert_process_images_hijack_called(batch_count=1) def test_process_images__2_batches_2_units__runs_2_batches(self): self.p.script_args = [ controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[ get_dummy_image(0), get_dummy_image(1), ], ), controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[ get_dummy_image(2), get_dummy_image(3), ], ), ] self.assert_process_images_hijack_called(batch_count=2) def test_process_images__3_batches_2_mixed_units__runs_3_batches(self): self.p.script_args = [ controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.BATCH, batch_images=[ get_dummy_image(0), get_dummy_image(1), get_dummy_image(2), ], ), controlnet.UiControlNetUnit( input_mode=batch_hijack.InputMode.SIMPLE, image=get_dummy_image(3), ), ] self.assert_process_images_hijack_called(batch_count=3) def get_dummy_image(name: Any = 0): return f'base64#{name}...'