Spaces:
Paused
Paused
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): | |
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) | |
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(list(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): | |
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 | |
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}...' | |
if __name__ == '__main__': | |
unittest.main() |