mirror of
https://github.com/AUTOMATIC1111/stable-diffusion-webui.git
synced 2024-06-07 21:20:49 +00:00
Merge pull request #4635 from mezotaken/master
CI tests with github-actions and some improvements to testing
This commit is contained in:
commit
8c8ad93bb5
31
.github/workflows/run_tests.yaml
vendored
Normal file
31
.github/workflows/run_tests.yaml
vendored
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
name: Run basic features tests on CPU with empty SD model
|
||||||
|
|
||||||
|
on:
|
||||||
|
- push
|
||||||
|
- pull_request
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout Code
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: Set up Python 3.10
|
||||||
|
uses: actions/setup-python@v4
|
||||||
|
with:
|
||||||
|
python-version: 3.10.6
|
||||||
|
- uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/.cache/pip
|
||||||
|
key: ${{ runner.os }}-pip-${{ hashFiles('**/requirements.txt') }}
|
||||||
|
restore-keys: ${{ runner.os }}-pip-
|
||||||
|
- name: Run tests
|
||||||
|
run: python launch.py --tests basic_features --no-half --disable-opt-split-attention --use-cpu all --skip-torch-cuda-test
|
||||||
|
- name: Upload main app stdout-stderr
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
if: always()
|
||||||
|
with:
|
||||||
|
name: stdout-stderr
|
||||||
|
path: |
|
||||||
|
test/stdout.txt
|
||||||
|
test/stderr.txt
|
38
launch.py
38
launch.py
@ -17,6 +17,19 @@ def extract_arg(args, name):
|
|||||||
return [x for x in args if x != name], name in args
|
return [x for x in args if x != name], name in args
|
||||||
|
|
||||||
|
|
||||||
|
def extract_opt(args, name):
|
||||||
|
opt = None
|
||||||
|
is_present = False
|
||||||
|
if name in args:
|
||||||
|
is_present = True
|
||||||
|
idx = args.index(name)
|
||||||
|
del args[idx]
|
||||||
|
if idx < len(args) and args[idx][0] != "-":
|
||||||
|
opt = args[idx]
|
||||||
|
del args[idx]
|
||||||
|
return args, is_present, opt
|
||||||
|
|
||||||
|
|
||||||
def run(command, desc=None, errdesc=None, custom_env=None):
|
def run(command, desc=None, errdesc=None, custom_env=None):
|
||||||
if desc is not None:
|
if desc is not None:
|
||||||
print(desc)
|
print(desc)
|
||||||
@ -151,12 +164,11 @@ def prepare_enviroment():
|
|||||||
blip_commit_hash = os.environ.get('BLIP_COMMIT_HASH', "48211a1594f1321b00f14c9f7a5b4813144b2fb9")
|
blip_commit_hash = os.environ.get('BLIP_COMMIT_HASH', "48211a1594f1321b00f14c9f7a5b4813144b2fb9")
|
||||||
|
|
||||||
sys.argv += shlex.split(commandline_args)
|
sys.argv += shlex.split(commandline_args)
|
||||||
test_argv = [x for x in sys.argv if x != '--tests']
|
|
||||||
|
|
||||||
sys.argv, skip_torch_cuda_test = extract_arg(sys.argv, '--skip-torch-cuda-test')
|
sys.argv, skip_torch_cuda_test = extract_arg(sys.argv, '--skip-torch-cuda-test')
|
||||||
sys.argv, reinstall_xformers = extract_arg(sys.argv, '--reinstall-xformers')
|
sys.argv, reinstall_xformers = extract_arg(sys.argv, '--reinstall-xformers')
|
||||||
sys.argv, update_check = extract_arg(sys.argv, '--update-check')
|
sys.argv, update_check = extract_arg(sys.argv, '--update-check')
|
||||||
sys.argv, run_tests = extract_arg(sys.argv, '--tests')
|
sys.argv, run_tests, test_dir = extract_opt(sys.argv, '--tests')
|
||||||
xformers = '--xformers' in sys.argv
|
xformers = '--xformers' in sys.argv
|
||||||
ngrok = '--ngrok' in sys.argv
|
ngrok = '--ngrok' in sys.argv
|
||||||
|
|
||||||
@ -221,24 +233,30 @@ def prepare_enviroment():
|
|||||||
exit(0)
|
exit(0)
|
||||||
|
|
||||||
if run_tests:
|
if run_tests:
|
||||||
tests(test_argv)
|
exitcode = tests(test_dir)
|
||||||
exit(0)
|
exit(exitcode)
|
||||||
|
|
||||||
|
|
||||||
def tests(argv):
|
def tests(test_dir):
|
||||||
if "--api" not in argv:
|
if "--api" not in sys.argv:
|
||||||
argv.append("--api")
|
sys.argv.append("--api")
|
||||||
|
if "--ckpt" not in sys.argv:
|
||||||
|
sys.argv.append("--ckpt")
|
||||||
|
sys.argv.append("./test/test_files/empty.pt")
|
||||||
|
if "--skip-torch-cuda-test" not in sys.argv:
|
||||||
|
sys.argv.append("--skip-torch-cuda-test")
|
||||||
|
|
||||||
print(f"Launching Web UI in another process for testing with arguments: {' '.join(argv[1:])}")
|
print(f"Launching Web UI in another process for testing with arguments: {' '.join(sys.argv[1:])}")
|
||||||
|
|
||||||
with open('test/stdout.txt', "w", encoding="utf8") as stdout, open('test/stderr.txt', "w", encoding="utf8") as stderr:
|
with open('test/stdout.txt', "w", encoding="utf8") as stdout, open('test/stderr.txt', "w", encoding="utf8") as stderr:
|
||||||
proc = subprocess.Popen([sys.executable, *argv], stdout=stdout, stderr=stderr)
|
proc = subprocess.Popen([sys.executable, *sys.argv], stdout=stdout, stderr=stderr)
|
||||||
|
|
||||||
import test.server_poll
|
import test.server_poll
|
||||||
test.server_poll.run_tests()
|
exitcode = test.server_poll.run_tests(proc, test_dir)
|
||||||
|
|
||||||
print(f"Stopping Web UI process with id {proc.pid}")
|
print(f"Stopping Web UI process with id {proc.pid}")
|
||||||
proc.kill()
|
proc.kill()
|
||||||
|
return exitcode
|
||||||
|
|
||||||
|
|
||||||
def start():
|
def start():
|
||||||
|
0
test/advanced_features/__init__.py
Normal file
0
test/advanced_features/__init__.py
Normal file
@ -11,8 +11,8 @@ class TestExtrasWorking(unittest.TestCase):
|
|||||||
"codeformer_visibility": 0,
|
"codeformer_visibility": 0,
|
||||||
"codeformer_weight": 0,
|
"codeformer_weight": 0,
|
||||||
"upscaling_resize": 2,
|
"upscaling_resize": 2,
|
||||||
"upscaling_resize_w": 512,
|
"upscaling_resize_w": 128,
|
||||||
"upscaling_resize_h": 512,
|
"upscaling_resize_h": 128,
|
||||||
"upscaling_crop": True,
|
"upscaling_crop": True,
|
||||||
"upscaler_1": "None",
|
"upscaler_1": "None",
|
||||||
"upscaler_2": "None",
|
"upscaler_2": "None",
|
47
test/advanced_features/txt2img_test.py
Normal file
47
test/advanced_features/txt2img_test.py
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
import unittest
|
||||||
|
import requests
|
||||||
|
|
||||||
|
|
||||||
|
class TestTxt2ImgWorking(unittest.TestCase):
|
||||||
|
def setUp(self):
|
||||||
|
self.url_txt2img = "http://localhost:7860/sdapi/v1/txt2img"
|
||||||
|
self.simple_txt2img = {
|
||||||
|
"enable_hr": False,
|
||||||
|
"denoising_strength": 0,
|
||||||
|
"firstphase_width": 0,
|
||||||
|
"firstphase_height": 0,
|
||||||
|
"prompt": "example prompt",
|
||||||
|
"styles": [],
|
||||||
|
"seed": -1,
|
||||||
|
"subseed": -1,
|
||||||
|
"subseed_strength": 0,
|
||||||
|
"seed_resize_from_h": -1,
|
||||||
|
"seed_resize_from_w": -1,
|
||||||
|
"batch_size": 1,
|
||||||
|
"n_iter": 1,
|
||||||
|
"steps": 3,
|
||||||
|
"cfg_scale": 7,
|
||||||
|
"width": 64,
|
||||||
|
"height": 64,
|
||||||
|
"restore_faces": False,
|
||||||
|
"tiling": False,
|
||||||
|
"negative_prompt": "",
|
||||||
|
"eta": 0,
|
||||||
|
"s_churn": 0,
|
||||||
|
"s_tmax": 0,
|
||||||
|
"s_tmin": 0,
|
||||||
|
"s_noise": 1,
|
||||||
|
"sampler_index": "Euler a"
|
||||||
|
}
|
||||||
|
|
||||||
|
def test_txt2img_with_restore_faces_performed(self):
|
||||||
|
self.simple_txt2img["restore_faces"] = True
|
||||||
|
self.assertEqual(requests.post(self.url_txt2img, json=self.simple_txt2img).status_code, 200)
|
||||||
|
|
||||||
|
|
||||||
|
class TestTxt2ImgCorrectness(unittest.TestCase):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
unittest.main()
|
0
test/basic_features/__init__.py
Normal file
0
test/basic_features/__init__.py
Normal file
@ -51,9 +51,5 @@ class TestImg2ImgWorking(unittest.TestCase):
|
|||||||
self.assertEqual(requests.post(self.url_img2img, json=self.simple_img2img).status_code, 200)
|
self.assertEqual(requests.post(self.url_img2img, json=self.simple_img2img).status_code, 200)
|
||||||
|
|
||||||
|
|
||||||
class TestImg2ImgCorrectness(unittest.TestCase):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
unittest.main()
|
unittest.main()
|
@ -49,26 +49,20 @@ class TestTxt2ImgWorking(unittest.TestCase):
|
|||||||
self.simple_txt2img["enable_hr"] = True
|
self.simple_txt2img["enable_hr"] = True
|
||||||
self.assertEqual(requests.post(self.url_txt2img, json=self.simple_txt2img).status_code, 200)
|
self.assertEqual(requests.post(self.url_txt2img, json=self.simple_txt2img).status_code, 200)
|
||||||
|
|
||||||
def test_txt2img_with_restore_faces_performed(self):
|
def test_txt2img_with_tiling_performed(self):
|
||||||
self.simple_txt2img["restore_faces"] = True
|
|
||||||
self.assertEqual(requests.post(self.url_txt2img, json=self.simple_txt2img).status_code, 200)
|
|
||||||
|
|
||||||
def test_txt2img_with_tiling_faces_performed(self):
|
|
||||||
self.simple_txt2img["tiling"] = True
|
self.simple_txt2img["tiling"] = True
|
||||||
self.assertEqual(requests.post(self.url_txt2img, json=self.simple_txt2img).status_code, 200)
|
self.assertEqual(requests.post(self.url_txt2img, json=self.simple_txt2img).status_code, 200)
|
||||||
|
|
||||||
def test_txt2img_with_vanilla_sampler_performed(self):
|
def test_txt2img_with_vanilla_sampler_performed(self):
|
||||||
self.simple_txt2img["sampler_index"] = "PLMS"
|
self.simple_txt2img["sampler_index"] = "PLMS"
|
||||||
self.assertEqual(requests.post(self.url_txt2img, json=self.simple_txt2img).status_code, 200)
|
self.assertEqual(requests.post(self.url_txt2img, json=self.simple_txt2img).status_code, 200)
|
||||||
|
self.simple_txt2img["sampler_index"] = "DDIM"
|
||||||
|
self.assertEqual(requests.post(self.url_txt2img, json=self.simple_txt2img).status_code, 200)
|
||||||
|
|
||||||
def test_txt2img_multiple_batches_performed(self):
|
def test_txt2img_multiple_batches_performed(self):
|
||||||
self.simple_txt2img["n_iter"] = 2
|
self.simple_txt2img["n_iter"] = 2
|
||||||
self.assertEqual(requests.post(self.url_txt2img, json=self.simple_txt2img).status_code, 200)
|
self.assertEqual(requests.post(self.url_txt2img, json=self.simple_txt2img).status_code, 200)
|
||||||
|
|
||||||
|
|
||||||
class TestTxt2ImgCorrectness(unittest.TestCase):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
unittest.main()
|
unittest.main()
|
@ -18,20 +18,6 @@ class UtilsTests(unittest.TestCase):
|
|||||||
def test_options_get(self):
|
def test_options_get(self):
|
||||||
self.assertEqual(requests.get(self.url_options).status_code, 200)
|
self.assertEqual(requests.get(self.url_options).status_code, 200)
|
||||||
|
|
||||||
def test_options_write(self):
|
|
||||||
response = requests.get(self.url_options)
|
|
||||||
self.assertEqual(response.status_code, 200)
|
|
||||||
|
|
||||||
pre_value = response.json()["send_seed"]
|
|
||||||
|
|
||||||
self.assertEqual(requests.post(self.url_options, json={"send_seed":not pre_value}).status_code, 200)
|
|
||||||
|
|
||||||
response = requests.get(self.url_options)
|
|
||||||
self.assertEqual(response.status_code, 200)
|
|
||||||
self.assertEqual(response.json()["send_seed"], not pre_value)
|
|
||||||
|
|
||||||
requests.post(self.url_options, json={"send_seed": pre_value})
|
|
||||||
|
|
||||||
def test_cmd_flags(self):
|
def test_cmd_flags(self):
|
||||||
self.assertEqual(requests.get(self.url_cmd_flags).status_code, 200)
|
self.assertEqual(requests.get(self.url_cmd_flags).status_code, 200)
|
||||||
|
|
||||||
@ -60,4 +46,8 @@ class UtilsTests(unittest.TestCase):
|
|||||||
self.assertEqual(requests.get(self.url_artist_categories).status_code, 200)
|
self.assertEqual(requests.get(self.url_artist_categories).status_code, 200)
|
||||||
|
|
||||||
def test_artists(self):
|
def test_artists(self):
|
||||||
self.assertEqual(requests.get(self.url_artists).status_code, 200)
|
self.assertEqual(requests.get(self.url_artists).status_code, 200)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
unittest.main()
|
@ -3,7 +3,7 @@ import requests
|
|||||||
import time
|
import time
|
||||||
|
|
||||||
|
|
||||||
def run_tests():
|
def run_tests(proc, test_dir):
|
||||||
timeout_threshold = 240
|
timeout_threshold = 240
|
||||||
start_time = time.time()
|
start_time = time.time()
|
||||||
while time.time()-start_time < timeout_threshold:
|
while time.time()-start_time < timeout_threshold:
|
||||||
@ -11,9 +11,14 @@ def run_tests():
|
|||||||
requests.head("http://localhost:7860/")
|
requests.head("http://localhost:7860/")
|
||||||
break
|
break
|
||||||
except requests.exceptions.ConnectionError:
|
except requests.exceptions.ConnectionError:
|
||||||
pass
|
if proc.poll() is not None:
|
||||||
if time.time()-start_time < timeout_threshold:
|
break
|
||||||
suite = unittest.TestLoader().discover('', pattern='*_test.py')
|
if proc.poll() is None:
|
||||||
|
if test_dir is None:
|
||||||
|
test_dir = ""
|
||||||
|
suite = unittest.TestLoader().discover(test_dir, pattern="*_test.py", top_level_dir="test")
|
||||||
result = unittest.TextTestRunner(verbosity=2).run(suite)
|
result = unittest.TextTestRunner(verbosity=2).run(suite)
|
||||||
|
return len(result.failures) + len(result.errors)
|
||||||
else:
|
else:
|
||||||
print("Launch unsuccessful")
|
print("Launch unsuccessful")
|
||||||
|
return 1
|
||||||
|
BIN
test/test_files/empty.pt
Normal file
BIN
test/test_files/empty.pt
Normal file
Binary file not shown.
Loading…
Reference in New Issue
Block a user